Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 : DNS utility library
4 : Copyright (C) Gerald (Jerry) Carter 2006.
5 : Copyright (C) Jeremy Allison 2007.
6 :
7 : This program is free software; you can redistribute it and/or modify
8 : it under the terms of the GNU General Public License as published by
9 : the Free Software Foundation; either version 3 of the License, or
10 : (at your option) any later version.
11 :
12 : This program is distributed in the hope that it will be useful,
13 : but WITHOUT ANY WARRANTY; without even the implied warranty of
14 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 : GNU General Public License for more details.
16 :
17 : You should have received a copy of the GNU General Public License
18 : along with this program. If not, see <http://www.gnu.org/licenses/>.
19 : */
20 :
21 : #include "includes.h"
22 : #include "lib/util/util_net.h"
23 : #include "lib/util/tsort.h"
24 : #include "librpc/gen_ndr/dns.h"
25 : #include "libcli/dns/dns_lookup.h"
26 : #include "lib/util/tevent_ntstatus.h"
27 : #include "dnsquery.h"
28 :
29 : /*********************************************************************
30 : Sort SRV record list based on weight and priority. See RFC 2782.
31 : *********************************************************************/
32 :
33 0 : static int dnssrvcmp( struct dns_rr_srv *a, struct dns_rr_srv *b )
34 : {
35 0 : if ( a->priority == b->priority ) {
36 :
37 : /* randomize entries with an equal weight and priority */
38 0 : if ( a->weight == b->weight )
39 0 : return 0;
40 :
41 : /* higher weights should be sorted lower */
42 0 : if ( a->weight > b->weight )
43 0 : return -1;
44 : else
45 0 : return 1;
46 : }
47 :
48 0 : if ( a->priority < b->priority )
49 0 : return -1;
50 :
51 0 : return 1;
52 : }
53 :
54 : struct ads_dns_lookup_srv_state {
55 : struct dns_rr_srv *srvs;
56 : size_t num_srvs;
57 : };
58 :
59 : static void ads_dns_lookup_srv_done(struct tevent_req *subreq);
60 :
61 58 : struct tevent_req *ads_dns_lookup_srv_send(TALLOC_CTX *mem_ctx,
62 : struct tevent_context *ev,
63 : const char *name)
64 : {
65 : struct tevent_req *req, *subreq;
66 : struct ads_dns_lookup_srv_state *state;
67 :
68 58 : req = tevent_req_create(mem_ctx, &state,
69 : struct ads_dns_lookup_srv_state);
70 58 : if (req == NULL) {
71 0 : return NULL;
72 : }
73 :
74 58 : subreq = dns_lookup_send(
75 : state,
76 : ev,
77 : NULL,
78 : name,
79 : DNS_QCLASS_IN,
80 : DNS_QTYPE_SRV);
81 :
82 58 : if (tevent_req_nomem(subreq, req)) {
83 0 : return tevent_req_post(req, ev);
84 : }
85 58 : tevent_req_set_callback(subreq, ads_dns_lookup_srv_done, req);
86 58 : return req;
87 : }
88 :
89 58 : static void ads_dns_lookup_srv_done(struct tevent_req *subreq)
90 : {
91 58 : struct tevent_req *req = tevent_req_callback_data(
92 : subreq, struct tevent_req);
93 58 : struct ads_dns_lookup_srv_state *state = tevent_req_data(
94 : req, struct ads_dns_lookup_srv_state);
95 : int ret;
96 : struct dns_name_packet *reply;
97 : uint16_t i, idx;
98 :
99 58 : ret = dns_lookup_recv(subreq, state, &reply);
100 58 : TALLOC_FREE(subreq);
101 58 : if (ret != 0) {
102 3 : tevent_req_nterror(req, map_nt_error_from_unix_common(ret));
103 3 : return;
104 : }
105 :
106 107 : for (i=0; i<reply->ancount; i++) {
107 52 : if (reply->answers[i].rr_type == DNS_QTYPE_SRV) {
108 : /* uint16_t can't wrap here. */
109 52 : state->num_srvs += 1;
110 : }
111 : }
112 :
113 55 : state->srvs = talloc_array(state, struct dns_rr_srv, state->num_srvs);
114 55 : if (tevent_req_nomem(state->srvs, req)) {
115 0 : return;
116 : }
117 :
118 55 : idx = 0;
119 :
120 107 : for (i=0; i<reply->ancount; i++) {
121 52 : struct dns_res_rec *an = &reply->answers[i];
122 52 : struct dns_rr_srv *dst = &state->srvs[idx];
123 : struct dns_srv_record *src;
124 :
125 52 : if (an->rr_type != DNS_QTYPE_SRV) {
126 0 : continue;
127 : }
128 52 : src = &an->rdata.srv_record;
129 :
130 52 : *dst = (struct dns_rr_srv) {
131 52 : .hostname = talloc_move(state->srvs, &src->target),
132 52 : .priority = src->priority,
133 52 : .weight = src->weight,
134 52 : .port = src->port,
135 : };
136 52 : idx += 1;
137 : }
138 :
139 55 : for (i=0; i<reply->arcount; i++) {
140 0 : struct dns_res_rec *ar = &reply->additional[i];
141 : struct sockaddr_storage addr;
142 : bool ok;
143 : size_t j;
144 :
145 0 : ok = dns_res_rec_get_sockaddr(ar, &addr);
146 0 : if (!ok) {
147 0 : continue;
148 : }
149 :
150 0 : for (j=0; j<state->num_srvs; j++) {
151 0 : struct dns_rr_srv *srv = &state->srvs[j];
152 : struct sockaddr_storage *tmp;
153 :
154 0 : if (strcmp(srv->hostname, ar->name) != 0) {
155 0 : continue;
156 : }
157 : /* uint16_t can't wrap here. */
158 0 : tmp = talloc_realloc(
159 : state->srvs,
160 : srv->ss_s,
161 : struct sockaddr_storage,
162 : srv->num_ips+1);
163 :
164 0 : if (tevent_req_nomem(tmp, req)) {
165 0 : return;
166 : }
167 0 : srv->ss_s = tmp;
168 :
169 0 : srv->ss_s[srv->num_ips] = addr;
170 0 : srv->num_ips += 1;
171 : }
172 : }
173 :
174 55 : TYPESAFE_QSORT(state->srvs, state->num_srvs, dnssrvcmp);
175 :
176 55 : tevent_req_done(req);
177 : }
178 :
179 58 : NTSTATUS ads_dns_lookup_srv_recv(struct tevent_req *req,
180 : TALLOC_CTX *mem_ctx,
181 : struct dns_rr_srv **srvs,
182 : size_t *num_srvs)
183 : {
184 58 : struct ads_dns_lookup_srv_state *state = tevent_req_data(
185 : req, struct ads_dns_lookup_srv_state);
186 : NTSTATUS status;
187 :
188 58 : if (tevent_req_is_nterror(req, &status)) {
189 3 : return status;
190 : }
191 55 : *srvs = talloc_move(mem_ctx, &state->srvs);
192 55 : *num_srvs = state->num_srvs;
193 55 : tevent_req_received(req);
194 55 : return NT_STATUS_OK;
195 : }
196 :
197 : /*********************************************************************
198 : Simple wrapper for a DNS SRV query
199 : *********************************************************************/
200 :
201 0 : NTSTATUS ads_dns_lookup_srv(TALLOC_CTX *ctx,
202 : const char *name,
203 : struct dns_rr_srv **dclist,
204 : size_t *numdcs)
205 : {
206 : struct tevent_context *ev;
207 : struct tevent_req *req;
208 0 : NTSTATUS status = NT_STATUS_NO_MEMORY;
209 0 : size_t num_srvs = 0;
210 :
211 0 : ev = samba_tevent_context_init(ctx);
212 0 : if (ev == NULL) {
213 0 : goto fail;
214 : }
215 0 : req = ads_dns_lookup_srv_send(ev, ev, name);
216 0 : if (req == NULL) {
217 0 : goto fail;
218 : }
219 0 : if (!tevent_req_poll_ntstatus(req, ev, &status)) {
220 0 : goto fail;
221 : }
222 0 : status = ads_dns_lookup_srv_recv(req, ctx, dclist, &num_srvs);
223 0 : if (NT_STATUS_IS_OK(status)) {
224 0 : *numdcs = num_srvs;
225 : }
226 0 : fail:
227 0 : TALLOC_FREE(ev);
228 0 : return status;
229 : }
230 :
231 : struct ads_dns_lookup_ns_state {
232 : struct dns_rr_ns *nss;
233 : size_t num_nss;
234 : };
235 :
236 : static void ads_dns_lookup_ns_done(struct tevent_req *subreq);
237 :
238 36 : struct tevent_req *ads_dns_lookup_ns_send(TALLOC_CTX *mem_ctx,
239 : struct tevent_context *ev,
240 : const char *name)
241 : {
242 : struct tevent_req *req, *subreq;
243 : struct ads_dns_lookup_ns_state *state;
244 :
245 36 : req = tevent_req_create(mem_ctx, &state,
246 : struct ads_dns_lookup_ns_state);
247 36 : if (req == NULL) {
248 0 : return NULL;
249 : }
250 :
251 36 : subreq = dns_lookup_send(state, ev, NULL, name, DNS_QCLASS_IN,
252 : DNS_QTYPE_NS);
253 36 : if (tevent_req_nomem(subreq, req)) {
254 0 : return tevent_req_post(req, ev);
255 : }
256 36 : tevent_req_set_callback(subreq, ads_dns_lookup_ns_done, req);
257 36 : return req;
258 : }
259 :
260 36 : static void ads_dns_lookup_ns_done(struct tevent_req *subreq)
261 : {
262 36 : struct tevent_req *req = tevent_req_callback_data(
263 : subreq, struct tevent_req);
264 36 : struct ads_dns_lookup_ns_state *state = tevent_req_data(
265 : req, struct ads_dns_lookup_ns_state);
266 : int ret;
267 : struct dns_name_packet *reply;
268 : uint16_t i, idx;
269 :
270 36 : ret = dns_lookup_recv(subreq, state, &reply);
271 36 : TALLOC_FREE(subreq);
272 36 : if (ret != 0) {
273 0 : tevent_req_nterror(req, map_nt_error_from_unix_common(ret));
274 0 : return;
275 : }
276 :
277 70 : for (i=0; i<reply->ancount; i++) {
278 34 : if (reply->answers[i].rr_type == DNS_QTYPE_NS) {
279 34 : state->num_nss += 1;
280 : }
281 : }
282 :
283 36 : state->nss = talloc_array(state, struct dns_rr_ns, state->num_nss);
284 36 : if (tevent_req_nomem(state->nss, req)) {
285 0 : return;
286 : }
287 :
288 36 : idx = 0;
289 :
290 70 : for (i=0; i<reply->ancount; i++) {
291 34 : struct dns_res_rec *an = &reply->answers[i];
292 :
293 34 : if (an->rr_type != DNS_QTYPE_NS) {
294 0 : continue;
295 : }
296 :
297 34 : state->nss[idx].hostname = talloc_move(state->nss,
298 : &an->rdata.ns_record);
299 34 : idx += 1;
300 : }
301 :
302 36 : for (i=0; i<reply->arcount; i++) {
303 0 : struct dns_res_rec *ar = &reply->additional[i];
304 : struct sockaddr_storage addr;
305 : bool ok;
306 : size_t j;
307 :
308 0 : ok = dns_res_rec_get_sockaddr(ar, &addr);
309 0 : if (!ok) {
310 0 : continue;
311 : }
312 :
313 0 : for (j=0; j<state->num_nss; j++) {
314 0 : struct dns_rr_ns *ns = &state->nss[j];
315 :
316 0 : if (strcmp(ns->hostname, ar->name) == 0) {
317 0 : ns->ss = addr;
318 : }
319 : }
320 : }
321 :
322 36 : tevent_req_done(req);
323 : }
324 :
325 36 : NTSTATUS ads_dns_lookup_ns_recv(struct tevent_req *req,
326 : TALLOC_CTX *mem_ctx,
327 : struct dns_rr_ns **nss,
328 : size_t *num_nss)
329 : {
330 36 : struct ads_dns_lookup_ns_state *state = tevent_req_data(
331 : req, struct ads_dns_lookup_ns_state);
332 : NTSTATUS status;
333 :
334 36 : if (tevent_req_is_nterror(req, &status)) {
335 0 : return status;
336 : }
337 36 : *nss = talloc_move(mem_ctx, &state->nss);
338 36 : *num_nss = state->num_nss;
339 36 : tevent_req_received(req);
340 36 : return NT_STATUS_OK;
341 : }
342 :
343 : /*********************************************************************
344 : Simple wrapper for a DNS NS query
345 : *********************************************************************/
346 :
347 36 : NTSTATUS ads_dns_lookup_ns(TALLOC_CTX *ctx,
348 : const char *dnsdomain,
349 : struct dns_rr_ns **nslist,
350 : size_t *numns)
351 : {
352 : struct tevent_context *ev;
353 : struct tevent_req *req;
354 36 : NTSTATUS status = NT_STATUS_NO_MEMORY;
355 36 : size_t num_ns = 0;
356 :
357 36 : ev = samba_tevent_context_init(ctx);
358 36 : if (ev == NULL) {
359 0 : goto fail;
360 : }
361 36 : req = ads_dns_lookup_ns_send(ev, ev, dnsdomain);
362 36 : if (req == NULL) {
363 0 : goto fail;
364 : }
365 36 : if (!tevent_req_poll_ntstatus(req, ev, &status)) {
366 0 : goto fail;
367 : }
368 36 : status = ads_dns_lookup_ns_recv(req, ctx, nslist, &num_ns);
369 36 : *numns = num_ns;
370 36 : fail:
371 36 : TALLOC_FREE(ev);
372 36 : return status;
373 : }
374 :
375 : /*********************************************************************
376 : Async A record lookup.
377 : *********************************************************************/
378 :
379 : struct ads_dns_lookup_a_state {
380 : uint8_t rcode;
381 : size_t num_names;
382 : char **hostnames;
383 : struct samba_sockaddr *addrs;
384 : };
385 :
386 : static void ads_dns_lookup_a_done(struct tevent_req *subreq);
387 :
388 54 : struct tevent_req *ads_dns_lookup_a_send(TALLOC_CTX *mem_ctx,
389 : struct tevent_context *ev,
390 : const char *name)
391 : {
392 54 : struct tevent_req *req = NULL, *subreq = NULL;
393 54 : struct ads_dns_lookup_a_state *state = NULL;
394 :
395 54 : req = tevent_req_create(mem_ctx, &state,
396 : struct ads_dns_lookup_a_state);
397 54 : if (req == NULL) {
398 0 : return NULL;
399 : }
400 :
401 54 : subreq = dns_lookup_send(
402 : state,
403 : ev,
404 : NULL,
405 : name,
406 : DNS_QCLASS_IN,
407 : DNS_QTYPE_A);
408 :
409 54 : if (tevent_req_nomem(subreq, req)) {
410 0 : return tevent_req_post(req, ev);
411 : }
412 54 : tevent_req_set_callback(subreq, ads_dns_lookup_a_done, req);
413 54 : return req;
414 : }
415 :
416 54 : static void ads_dns_lookup_a_done(struct tevent_req *subreq)
417 : {
418 54 : struct tevent_req *req = tevent_req_callback_data(
419 : subreq, struct tevent_req);
420 54 : struct ads_dns_lookup_a_state *state = tevent_req_data(
421 : req, struct ads_dns_lookup_a_state);
422 : int ret;
423 54 : struct dns_name_packet *reply = NULL;
424 : uint16_t i;
425 :
426 54 : ret = dns_lookup_recv(subreq, state, &reply);
427 54 : TALLOC_FREE(subreq);
428 54 : if (ret != 0) {
429 0 : tevent_req_nterror(req, map_nt_error_from_unix_common(ret));
430 0 : return;
431 : }
432 :
433 54 : state->rcode = (reply->operation & DNS_RCODE);
434 54 : if (state->rcode != DNS_RCODE_OK) {
435 : /* Don't bother looking for answers. */
436 0 : tevent_req_done(req);
437 0 : return;
438 : }
439 :
440 : /*
441 : * We don't care about CNAME answers here. We're
442 : * just wanting an async name -> IPv4 lookup.
443 : */
444 108 : for (i = 0; i < reply->ancount; i++) {
445 54 : if (reply->answers[i].rr_type == DNS_QTYPE_A) {
446 54 : state->num_names += 1;
447 : }
448 : }
449 :
450 54 : state->hostnames = talloc_zero_array(state,
451 : char *,
452 : state->num_names);
453 54 : if (tevent_req_nomem(state->hostnames, req)) {
454 0 : return;
455 : }
456 54 : state->addrs = talloc_zero_array(state,
457 : struct samba_sockaddr,
458 : state->num_names);
459 54 : if (tevent_req_nomem(state->addrs, req)) {
460 0 : return;
461 : }
462 :
463 54 : state->num_names = 0;
464 :
465 108 : for (i = 0; i < reply->ancount; i++) {
466 : bool ok;
467 54 : struct sockaddr_storage ss = {0};
468 54 : struct dns_res_rec *an = &reply->answers[i];
469 :
470 54 : if (an->rr_type != DNS_QTYPE_A) {
471 0 : continue;
472 : }
473 54 : if (an->name == NULL) {
474 : /* Can this happen? */
475 0 : continue;
476 : }
477 54 : if (an->rdata.ipv4_record == NULL) {
478 : /* Can this happen? */
479 0 : continue;
480 : }
481 54 : ok = dns_res_rec_get_sockaddr(an,
482 : &ss);
483 54 : if (!ok) {
484 0 : continue;
485 : }
486 54 : if (is_zero_addr(&ss)) {
487 0 : continue;
488 : }
489 54 : state->addrs[state->num_names].u.ss = ss;
490 54 : state->addrs[state->num_names].sa_socklen =
491 : sizeof(struct sockaddr_in);
492 108 : state->hostnames[state->num_names] = talloc_strdup(
493 54 : state->hostnames,
494 : an->name);
495 54 : if (tevent_req_nomem(state->hostnames[state->num_names], req)) {
496 0 : return;
497 : }
498 54 : state->num_names += 1;
499 : }
500 :
501 54 : tevent_req_done(req);
502 : }
503 :
504 54 : NTSTATUS ads_dns_lookup_a_recv(struct tevent_req *req,
505 : TALLOC_CTX *mem_ctx,
506 : uint8_t *rcode_out,
507 : size_t *num_names_out,
508 : char ***hostnames_out,
509 : struct samba_sockaddr **addrs_out)
510 : {
511 54 : struct ads_dns_lookup_a_state *state = tevent_req_data(
512 : req, struct ads_dns_lookup_a_state);
513 : NTSTATUS status;
514 :
515 54 : if (tevent_req_is_nterror(req, &status)) {
516 0 : return status;
517 : }
518 54 : if (rcode_out != NULL) {
519 : /*
520 : * If we got no names, an upper layer may
521 : * want to print a debug message.
522 : */
523 52 : *rcode_out = state->rcode;
524 : }
525 54 : if (hostnames_out != NULL) {
526 54 : *hostnames_out = talloc_move(mem_ctx,
527 : &state->hostnames);
528 : }
529 54 : if (addrs_out != NULL) {
530 54 : *addrs_out = talloc_move(mem_ctx,
531 : &state->addrs);
532 : }
533 54 : *num_names_out = state->num_names;
534 54 : tevent_req_received(req);
535 54 : return NT_STATUS_OK;
536 : }
537 :
538 : /*********************************************************************
539 : Simple wrapper for a DNS A query
540 : *********************************************************************/
541 :
542 2 : NTSTATUS ads_dns_lookup_a(TALLOC_CTX *ctx,
543 : const char *name_in,
544 : size_t *num_names_out,
545 : char ***hostnames_out,
546 : struct samba_sockaddr **addrs_out)
547 : {
548 : struct tevent_context *ev;
549 : struct tevent_req *req;
550 2 : NTSTATUS status = NT_STATUS_NO_MEMORY;
551 :
552 2 : ev = samba_tevent_context_init(ctx);
553 2 : if (ev == NULL) {
554 0 : goto fail;
555 : }
556 2 : req = ads_dns_lookup_a_send(ev, ev, name_in);
557 2 : if (req == NULL) {
558 0 : goto fail;
559 : }
560 2 : if (!tevent_req_poll_ntstatus(req, ev, &status)) {
561 0 : goto fail;
562 : }
563 : /*
564 : * Sychronous doesn't need to care about the rcode or
565 : * a copy of the name_in.
566 : */
567 2 : status = ads_dns_lookup_a_recv(req,
568 : ctx,
569 : NULL,
570 : num_names_out,
571 : hostnames_out,
572 : addrs_out);
573 2 : fail:
574 2 : TALLOC_FREE(ev);
575 2 : return status;
576 : }
577 :
578 : #if defined(HAVE_IPV6)
579 : /*********************************************************************
580 : Async AAAA record lookup.
581 : *********************************************************************/
582 :
583 : struct ads_dns_lookup_aaaa_state {
584 : uint8_t rcode;
585 : size_t num_names;
586 : char **hostnames;
587 : struct samba_sockaddr *addrs;
588 : };
589 :
590 : static void ads_dns_lookup_aaaa_done(struct tevent_req *subreq);
591 :
592 54 : struct tevent_req *ads_dns_lookup_aaaa_send(TALLOC_CTX *mem_ctx,
593 : struct tevent_context *ev,
594 : const char *name)
595 : {
596 54 : struct tevent_req *req, *subreq = NULL;
597 54 : struct ads_dns_lookup_aaaa_state *state = NULL;
598 :
599 54 : req = tevent_req_create(mem_ctx, &state,
600 : struct ads_dns_lookup_aaaa_state);
601 54 : if (req == NULL) {
602 0 : return NULL;
603 : }
604 :
605 54 : subreq = dns_lookup_send(
606 : state,
607 : ev,
608 : NULL,
609 : name,
610 : DNS_QCLASS_IN,
611 : DNS_QTYPE_AAAA);
612 :
613 54 : if (tevent_req_nomem(subreq, req)) {
614 0 : return tevent_req_post(req, ev);
615 : }
616 54 : tevent_req_set_callback(subreq, ads_dns_lookup_aaaa_done, req);
617 54 : return req;
618 : }
619 :
620 54 : static void ads_dns_lookup_aaaa_done(struct tevent_req *subreq)
621 : {
622 54 : struct tevent_req *req = tevent_req_callback_data(
623 : subreq, struct tevent_req);
624 54 : struct ads_dns_lookup_aaaa_state *state = tevent_req_data(
625 : req, struct ads_dns_lookup_aaaa_state);
626 : int ret;
627 54 : struct dns_name_packet *reply = NULL;
628 : uint16_t i;
629 :
630 54 : ret = dns_lookup_recv(subreq, state, &reply);
631 54 : TALLOC_FREE(subreq);
632 54 : if (ret != 0) {
633 0 : tevent_req_nterror(req, map_nt_error_from_unix_common(ret));
634 0 : return;
635 : }
636 :
637 54 : state->rcode = (reply->operation & DNS_RCODE);
638 54 : if (state->rcode != DNS_RCODE_OK) {
639 : /* Don't bother looking for answers. */
640 0 : tevent_req_done(req);
641 0 : return;
642 : }
643 :
644 : /*
645 : * We don't care about CNAME answers here. We're
646 : * just wanting an async name -> IPv6 lookup.
647 : */
648 108 : for (i = 0; i < reply->ancount; i++) {
649 54 : if (reply->answers[i].rr_type == DNS_QTYPE_AAAA) {
650 54 : state->num_names += 1;
651 : }
652 : }
653 :
654 54 : state->hostnames = talloc_zero_array(state,
655 : char *,
656 : state->num_names);
657 54 : if (tevent_req_nomem(state->hostnames, req)) {
658 0 : return;
659 : }
660 54 : state->addrs = talloc_zero_array(state,
661 : struct samba_sockaddr,
662 : state->num_names);
663 54 : if (tevent_req_nomem(state->addrs, req)) {
664 0 : return;
665 : }
666 :
667 54 : state->num_names = 0;
668 :
669 108 : for (i = 0; i < reply->ancount; i++) {
670 : bool ok;
671 54 : struct sockaddr_storage ss = {0};
672 54 : struct dns_res_rec *an = &reply->answers[i];
673 :
674 54 : if (an->rr_type != DNS_QTYPE_AAAA) {
675 0 : continue;
676 : }
677 54 : if (an->name == NULL) {
678 : /* Can this happen? */
679 0 : continue;
680 : }
681 54 : if (an->rdata.ipv6_record == NULL) {
682 : /* Can this happen? */
683 0 : continue;
684 : }
685 54 : ok = dns_res_rec_get_sockaddr(an,
686 : &ss);
687 54 : if (!ok) {
688 0 : continue;
689 : }
690 54 : if (is_zero_addr(&ss)) {
691 0 : continue;
692 : }
693 54 : state->addrs[state->num_names].u.ss = ss;
694 54 : state->addrs[state->num_names].sa_socklen =
695 : sizeof(struct sockaddr_in6);
696 :
697 108 : state->hostnames[state->num_names] = talloc_strdup(
698 54 : state->hostnames,
699 : an->name);
700 54 : if (tevent_req_nomem(state->hostnames[state->num_names], req)) {
701 0 : return;
702 : }
703 54 : state->num_names += 1;
704 : }
705 :
706 54 : tevent_req_done(req);
707 : }
708 :
709 54 : NTSTATUS ads_dns_lookup_aaaa_recv(struct tevent_req *req,
710 : TALLOC_CTX *mem_ctx,
711 : uint8_t *rcode_out,
712 : size_t *num_names_out,
713 : char ***hostnames_out,
714 : struct samba_sockaddr **addrs_out)
715 : {
716 54 : struct ads_dns_lookup_aaaa_state *state = tevent_req_data(
717 : req, struct ads_dns_lookup_aaaa_state);
718 : NTSTATUS status;
719 :
720 54 : if (tevent_req_is_nterror(req, &status)) {
721 0 : return status;
722 : }
723 54 : if (rcode_out != NULL) {
724 : /*
725 : * If we got no names, an upper layer may
726 : * want to print a debug message.
727 : */
728 52 : *rcode_out = state->rcode;
729 : }
730 54 : if (hostnames_out != NULL) {
731 54 : *hostnames_out = talloc_move(mem_ctx,
732 : &state->hostnames);
733 : }
734 54 : if (addrs_out != NULL) {
735 54 : *addrs_out = talloc_move(mem_ctx,
736 : &state->addrs);
737 : }
738 54 : *num_names_out = state->num_names;
739 54 : tevent_req_received(req);
740 54 : return NT_STATUS_OK;
741 : }
742 :
743 : /*********************************************************************
744 : Simple wrapper for a DNS AAAA query
745 : *********************************************************************/
746 :
747 2 : NTSTATUS ads_dns_lookup_aaaa(TALLOC_CTX *ctx,
748 : const char *name_in,
749 : size_t *num_names_out,
750 : char ***hostnames_out,
751 : struct samba_sockaddr **addrs_out)
752 : {
753 2 : struct tevent_context *ev = NULL;
754 2 : struct tevent_req *req = NULL;
755 2 : NTSTATUS status = NT_STATUS_NO_MEMORY;
756 :
757 2 : ev = samba_tevent_context_init(ctx);
758 2 : if (ev == NULL) {
759 0 : goto fail;
760 : }
761 2 : req = ads_dns_lookup_aaaa_send(ev, ev, name_in);
762 2 : if (req == NULL) {
763 0 : goto fail;
764 : }
765 2 : if (!tevent_req_poll_ntstatus(req, ev, &status)) {
766 0 : goto fail;
767 : }
768 : /*
769 : * Sychronous doesn't need to care about the rcode or
770 : * a copy of the name_in.
771 : */
772 2 : status = ads_dns_lookup_aaaa_recv(req,
773 : ctx,
774 : NULL,
775 : num_names_out,
776 : hostnames_out,
777 : addrs_out);
778 2 : fail:
779 2 : TALLOC_FREE(ev);
780 2 : return status;
781 : }
782 : #endif
|