2 * Copyright (C) 2004, 2005, 2007, 2009, 2012, 2014 Internet Systems Consortium, Inc. ("ISC")
3 * Copyright (C) 1999-2003 Internet Software Consortium.
5 * Permission to use, copy, modify, and/or distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
10 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
11 * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
12 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
13 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
14 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
15 * PERFORMANCE OF THIS SOFTWARE.
18 /* $Id: getipnode.c,v 1.47 2009/09/01 23:47:45 tbox Exp $ */
23 * These functions perform thread safe, protocol independent
24 * nodename-to-address and address-to-nodename translation as defined in
25 * RFC2553. This use a struct hostent which is defined in namedb.h:
29 * char *h_name; // official name of host
30 * char **h_aliases; // alias list
31 * int h_addrtype; // host address type
32 * int h_length; // length of address
33 * char **h_addr_list; // list of addresses from name server
35 * #define h_addr h_addr_list[0] // address, for backward compatibility
38 * The members of this structure are:
41 * The official (canonical) name of the host.
44 * A NULL-terminated array of alternate names (nicknames) for the
48 * The type of address being returned - usually PF_INET or
52 * The length of the address in bytes.
55 * A NULL terminated array of network addresses for the host. Host
56 * addresses are returned in network byte order.
58 * lwres_getipnodebyname() looks up addresses of protocol family af for
59 * the hostname name. The flags parameter contains ORed flag bits to
60 * specify the types of addresses that are searched for, and the types of
61 * addresses that are returned. The flag bits are:
64 * This is used with an af of #AF_INET6, and causes IPv4 addresses
65 * to be returned as IPv4-mapped IPv6 addresses.
68 * This is used with an af of #AF_INET6, and causes all known
69 * addresses (IPv6 and IPv4) to be returned. If #AI_V4MAPPED is
70 * also set, the IPv4 addresses are return as mapped IPv6
74 * Only return an IPv6 or IPv4 address if here is an active
75 * network interface of that type. This is not currently
76 * implemented in the BIND 9 lightweight resolver, and the flag is
80 * This default sets the #AI_V4MAPPED and #AI_ADDRCONFIG flag bits.
82 * lwres_getipnodebyaddr() performs a reverse lookup of address src which
83 * is len bytes long. af denotes the protocol family, typically PF_INET
86 * lwres_freehostent() releases all the memory associated with the struct
87 * hostent pointer. Any memory allocated for the h_name, h_addr_list
88 * and h_aliases is freed, as is the memory for the hostent structure
91 * \section getipnode_return Return Values
93 * If an error occurs, lwres_getipnodebyname() and
94 * lwres_getipnodebyaddr() set *error_num to an appropriate error code
95 * and the function returns a NULL pointer. The error codes and their
96 * meanings are defined in \link netdb.h <lwres/netdb.h>\endlink:
98 * \li #HOST_NOT_FOUND:
99 * No such host is known.
102 * The server recognised the request and the name but no address
103 * is available. Another type of request to the name server for
104 * the domain might return an answer.
107 * A temporary and possibly transient error occurred, such as a
108 * failure of a server to respond. The request may succeed if
112 * An unexpected failure occurred, and retrying the request is
115 * lwres_hstrerror() translates these error codes to suitable error
118 * \section getipnode_see See Also
120 * getaddrinfo.c, gethost.c, getnameinfo.c, herror.c, RFC2553
130 #include <lwres/lwres.h>
131 #include <lwres/net.h>
132 #include <lwres/netdb.h> /* XXX #include <netdb.h> */
134 #include "assert_p.h"
143 #ifdef LWRES_PLATFORM_NEEDIN6ADDRANY
144 LIBLWRES_EXTERNAL_DATA const struct in6_addr in6addr_any = IN6ADDR_ANY_INIT;
147 #ifndef IN6_IS_ADDR_V4COMPAT
148 static const unsigned char in6addr_compat[12] = {
149 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
151 #define IN6_IS_ADDR_V4COMPAT(x) (!memcmp((x)->s6_addr, in6addr_compat, 12) && \
152 ((x)->s6_addr[12] != 0 || \
153 (x)->s6_addr[13] != 0 || \
154 (x)->s6_addr[14] != 0 || \
155 ((x)->s6_addr[15] != 0 && \
156 (x)->s6_addr[15] != 1)))
158 #ifndef IN6_IS_ADDR_V4MAPPED
159 #define IN6_IS_ADDR_V4MAPPED(x) (!memcmp((x)->s6_addr, in6addr_mapped, 12))
162 static const unsigned char in6addr_mapped[12] = {
163 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff
167 *** Forward declarations.
171 scan_interfaces(int *, int *);
173 static struct hostent *
174 copyandmerge(struct hostent *, struct hostent *, int, int *);
176 static struct hostent *
177 hostfromaddr(lwres_gnbaresponse_t *addr, int af, const void *src);
179 static struct hostent *
180 hostfromname(lwres_gabnresponse_t *name, int af);
183 *** Public functions.
187 * AI_V4MAPPED + AF_INET6
188 * If no IPv6 address then a query for IPv4 and map returned values.
190 * AI_ALL + AI_V4MAPPED + AF_INET6
191 * Return IPv6 and IPv4 mapped.
194 * Only return IPv6 / IPv4 address if there is an interface of that
199 lwres_getipnodebyname(const char *name, int af, int flags, int *error_num) {
200 int have_v4 = 1, have_v6 = 1;
203 struct hostent he, *he1 = NULL, *he2 = NULL, *he3 = NULL;
206 lwres_context_t *lwrctx = NULL;
207 lwres_gabnresponse_t *by = NULL;
211 * If we care about active interfaces then check.
213 if ((flags & AI_ADDRCONFIG) != 0)
214 if (scan_interfaces(&have_v4, &have_v6) == -1) {
215 *error_num = NO_RECOVERY;
219 /* Check for literal address. */
220 if ((v4 = lwres_net_pton(AF_INET, name, &in4)) != 1)
221 v6 = lwres_net_pton(AF_INET6, name, &in6);
224 * Impossible combination?
226 if ((af == AF_INET6 && (flags & AI_V4MAPPED) == 0 && v4 == 1) ||
227 (af == AF_INET && v6 == 1) ||
228 (have_v4 == 0 && v4 == 1) ||
229 (have_v6 == 0 && v6 == 1) ||
230 (have_v4 == 0 && af == AF_INET) ||
231 (have_v6 == 0 && af == AF_INET6 &&
232 (((flags & AI_V4MAPPED) != 0 && have_v4) ||
233 (flags & AI_V4MAPPED) == 0))) {
234 *error_num = HOST_NOT_FOUND;
241 if (v4 == 1 || v6 == 1) {
244 char mappedname[sizeof("::ffff:123.123.123.123")];
246 const char *const_name;
251 if (v4 == 1 && af == AF_INET6) {
252 strcpy(mappedname, "::ffff:");
253 lwres_net_ntop(AF_INET, (char *)&in4,
254 mappedname + sizeof("::ffff:") - 1,
255 sizeof(mappedname) - sizeof("::ffff:")
257 he.h_name = mappedname;
259 he.h_name = u.deconst_name;
260 he.h_addr_list = addr_list;
261 he.h_addr_list[0] = (v4 == 1) ? (char *)&in4 : (char *)&in6;
262 he.h_addr_list[1] = NULL;
263 he.h_aliases = aliases;
264 he.h_aliases[0] = NULL;
265 he.h_length = (v4 == 1) ? INADDRSZ : IN6ADDRSZ;
266 he.h_addrtype = (v4 == 1) ? AF_INET : AF_INET6;
267 return (copyandmerge(&he, NULL, af, error_num));
270 n = lwres_context_create(&lwrctx, NULL, NULL, NULL, 0);
272 *error_num = NO_RECOVERY;
275 (void) lwres_conf_parse(lwrctx, lwres_resolv_conf);
276 tmp_err = NO_RECOVERY;
277 if (have_v6 && af == AF_INET6) {
278 n = lwres_getaddrsbyname(lwrctx, name, LWRES_ADDRTYPE_V6, &by);
280 he1 = hostfromname(by, AF_INET6);
281 lwres_gabnresponse_free(lwrctx, &by);
283 *error_num = NO_RECOVERY;
287 if (n == LWRES_R_NOTFOUND)
288 tmp_err = HOST_NOT_FOUND;
290 *error_num = NO_RECOVERY;
298 (af == AF_INET6 && (flags & AI_V4MAPPED) != 0 &&
299 (he1 == NULL || (flags & AI_ALL) != 0)))) {
300 n = lwres_getaddrsbyname(lwrctx, name, LWRES_ADDRTYPE_V4, &by);
302 he2 = hostfromname(by, AF_INET);
303 lwres_gabnresponse_free(lwrctx, &by);
305 *error_num = NO_RECOVERY;
308 } else if (he1 == NULL) {
309 if (n == LWRES_R_NOTFOUND)
310 *error_num = HOST_NOT_FOUND;
312 *error_num = NO_RECOVERY;
316 *error_num = tmp_err;
318 he3 = copyandmerge(he1, he2, af, error_num);
322 lwres_freehostent(he1);
324 lwres_freehostent(he2);
325 if (lwrctx != NULL) {
326 lwres_conf_clear(lwrctx);
327 lwres_context_destroy(&lwrctx);
332 /*% performs a reverse lookup of address src which is len bytes long. af denotes the protocol family, typically #PF_INET or PF_INET6. */
334 lwres_getipnodebyaddr(const void *src, size_t len, int af, int *error_num) {
335 struct hostent *he1, *he2;
336 lwres_context_t *lwrctx = NULL;
337 lwres_gnbaresponse_t *by = NULL;
341 struct in6_addr *in6;
348 *error_num = NO_RECOVERY;
354 if (len != (unsigned int)INADDRSZ) {
355 *error_num = NO_RECOVERY;
360 if (len != (unsigned int)IN6ADDRSZ) {
361 *error_num = NO_RECOVERY;
366 *error_num = NO_RECOVERY;
371 * The de-"const"-ing game is done because at least one
372 * vendor's system (RedHat 6.0) defines the IN6_IS_ADDR_*
373 * macros in such a way that they discard the const with
374 * internal casting, and gcc ends up complaining. Rather
375 * than replacing their own (possibly optimized) definitions
376 * with our own, cleanly discarding the const is the easiest
382 * Look up IPv4 and IPv4 mapped/compatible addresses.
384 if ((af == AF_INET6 && IN6_IS_ADDR_V4COMPAT(u.in6)) ||
385 (af == AF_INET6 && IN6_IS_ADDR_V4MAPPED(u.in6)) ||
387 const unsigned char *cp = src;
391 n = lwres_context_create(&lwrctx, NULL, NULL, NULL, 0);
392 if (n == LWRES_R_SUCCESS)
393 (void) lwres_conf_parse(lwrctx, lwres_resolv_conf);
394 if (n == LWRES_R_SUCCESS)
395 n = lwres_getnamebyaddr(lwrctx, LWRES_ADDRTYPE_V4,
397 if (n != LWRES_R_SUCCESS) {
398 lwres_conf_clear(lwrctx);
399 lwres_context_destroy(&lwrctx);
400 if (n == LWRES_R_NOTFOUND)
401 *error_num = HOST_NOT_FOUND;
403 *error_num = NO_RECOVERY;
406 he1 = hostfromaddr(by, AF_INET, cp);
407 lwres_gnbaresponse_free(lwrctx, &by);
408 lwres_conf_clear(lwrctx);
409 lwres_context_destroy(&lwrctx);
414 * Convert from AF_INET to AF_INET6.
416 he2 = copyandmerge(he1, NULL, af, error_num);
417 lwres_freehostent(he1);
421 * Restore original address.
423 memmove(he2->h_addr, src, len);
428 * Lookup IPv6 address.
430 if (memcmp(src, &in6addr_any, IN6ADDRSZ) == 0) {
431 *error_num = HOST_NOT_FOUND;
435 n = lwres_context_create(&lwrctx, NULL, NULL, NULL, 0);
436 if (n == LWRES_R_SUCCESS)
437 (void) lwres_conf_parse(lwrctx, lwres_resolv_conf);
438 if (n == LWRES_R_SUCCESS)
439 n = lwres_getnamebyaddr(lwrctx, LWRES_ADDRTYPE_V6, IN6ADDRSZ,
442 lwres_conf_clear(lwrctx);
443 lwres_context_destroy(&lwrctx);
445 if (n == LWRES_R_NOTFOUND)
446 *error_num = HOST_NOT_FOUND;
448 *error_num = NO_RECOVERY;
453 he1 = hostfromaddr(by, AF_INET6, src);
454 lwres_gnbaresponse_free(lwrctx, &by);
456 *error_num = NO_RECOVERY;
457 lwres_conf_clear(lwrctx);
458 lwres_context_destroy(&lwrctx);
462 /*% releases all the memory associated with the struct hostent pointer */
464 lwres_freehostent(struct hostent *he) {
474 cpp = he->h_addr_list;
475 while (*cpp != NULL) {
483 while (*cpp != NULL) {
490 free(he->h_addr_list);
499 * Scan the interface table and set have_v4 and have_v6 depending
500 * upon whether there are IPv4 and IPv6 interface addresses.
507 #if defined(SIOCGLIFCONF) && defined(SIOCGLIFADDR) && \
508 !defined(IRIX_EMUL_IOCTL_SIOCGIFCONF)
511 #define lifc_len iflc_len
512 #define lifc_buf iflc_buf
513 #define lifc_req iflc_req
514 #define LIFCONF if_laddrconf
516 #define ISC_HAVE_LIFC_FAMILY 1
517 #define ISC_HAVE_LIFC_FLAGS 1
518 #define LIFCONF lifconf
522 #define lifr_addr iflr_addr
523 #define lifr_name iflr_name
524 #define lifr_dstaddr iflr_dstaddr
525 #define lifr_flags iflr_flags
526 #define ss_family sa_family
527 #define LIFREQ if_laddrreq
529 #define LIFREQ lifreq
533 scan_interfaces6(int *have_v4, int *have_v6) {
535 struct LIFREQ lifreq;
538 char *buf = NULL, *cp, *cplim;
539 static unsigned int bufsiz = 4095;
543 * Set to zero. Used as loop terminators below.
545 *have_v4 = *have_v6 = 0;
548 * Get interface list from system.
550 if ((s = socket(AF_INET6, SOCK_DGRAM, 0)) == -1)
554 * Grow buffer until large enough to contain all interface
558 buf = malloc(bufsiz);
561 #ifdef ISC_HAVE_LIFC_FAMILY
562 lifc.lifc_family = AF_UNSPEC; /* request all families */
564 #ifdef ISC_HAVE_LIFC_FLAGS
567 lifc.lifc_len = bufsiz;
569 if ((n = ioctl(s, SIOCGLIFCONF, (char *)&lifc)) != -1) {
571 * Some OS's just return what will fit rather
572 * than set EINVAL if the buffer is too small
573 * to fit all the interfaces in. If
574 * lifc.lifc_len is too near to the end of the
575 * buffer we will grow it just in case and
578 if (lifc.lifc_len + 2 * sizeof(lifreq) < bufsiz)
581 if ((n == -1) && errno != EINVAL)
584 if (bufsiz > 1000000)
592 * Parse system's interface list.
594 cplim = buf + lifc.lifc_len; /* skip over if's with big ifr_addr's */
596 (*have_v4 == 0 || *have_v6 == 0) && cp < cplim;
598 memmove(&lifreq, cp, sizeof(lifreq));
599 #ifdef LWRES_PLATFORM_HAVESALEN
600 #ifdef FIX_ZERO_SA_LEN
601 if (lifreq.lifr_addr.sa_len == 0)
602 lifreq.lifr_addr.sa_len = 16;
604 #ifdef HAVE_MINIMUM_IFREQ
605 cpsize = sizeof(lifreq);
606 if (lifreq.lifr_addr.sa_len > sizeof(struct sockaddr))
607 cpsize += (int)lifreq.lifr_addr.sa_len -
608 (int)(sizeof(struct sockaddr));
610 cpsize = sizeof(lifreq.lifr_name) + lifreq.lifr_addr.sa_len;
611 #endif /* HAVE_MINIMUM_IFREQ */
612 #elif defined SIOCGIFCONF_ADDR
613 cpsize = sizeof(lifreq);
615 cpsize = sizeof(lifreq.lifr_name);
616 /* XXX maybe this should be a hard error? */
617 if (ioctl(s, SIOCGLIFADDR, (char *)&lifreq) < 0)
620 switch (lifreq.lifr_addr.ss_family) {
624 &((struct sockaddr_in *)
625 &lifreq.lifr_addr)->sin_addr,
627 if (in4.s_addr == INADDR_ANY)
629 n = ioctl(s, SIOCGLIFFLAGS, (char *)&lifreq);
632 if ((lifreq.lifr_flags & IFF_UP) == 0)
640 &((struct sockaddr_in6 *)
641 &lifreq.lifr_addr)->sin6_addr,
643 if (memcmp(&in6, &in6addr_any,
646 n = ioctl(s, SIOCGLIFFLAGS, (char *)&lifreq);
649 if ((lifreq.lifr_flags & IFF_UP) == 0)
670 scan_interfaces(int *have_v4, int *have_v6) {
671 #if !defined(SIOCGIFCONF) || !defined(SIOCGIFADDR)
672 *have_v4 = *have_v6 = 1;
677 char _pad[256]; /* leave space for IPv6 addresses */
682 char *buf = NULL, *cp, *cplim;
683 static unsigned int bufsiz = 4095;
690 #if defined(SIOCGLIFCONF) && defined(SIOCGLIFADDR) && \
691 !defined(IRIX_EMUL_IOCTL_SIOCGIFCONF)
693 * Try to scan the interfaces using IPv6 ioctls().
695 if (!scan_interfaces6(have_v4, have_v6)) {
704 * Set to zero. Used as loop terminators below.
706 *have_v4 = *have_v6 = 0;
709 * Get interface list from system.
711 if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
715 * Grow buffer until large enough to contain all interface
719 buf = malloc(bufsiz);
722 ifc.ifc_len = bufsiz;
724 #ifdef IRIX_EMUL_IOCTL_SIOCGIFCONF
726 * This is a fix for IRIX OS in which the call to ioctl with
727 * the flag SIOCGIFCONF may not return an entry for all the
728 * interfaces like most flavors of Unix.
730 if (emul_ioctl(&ifc) >= 0)
733 if ((n = ioctl(s, SIOCGIFCONF, (char *)&ifc)) != -1) {
735 * Some OS's just return what will fit rather
736 * than set EINVAL if the buffer is too small
737 * to fit all the interfaces in. If
738 * ifc.ifc_len is too near to the end of the
739 * buffer we will grow it just in case and
742 if (ifc.ifc_len + 2 * sizeof(u.ifreq) < bufsiz)
746 if ((n == -1) && errno != EINVAL)
749 if (bufsiz > 1000000)
757 * Parse system's interface list.
759 cplim = buf + ifc.ifc_len; /* skip over if's with big ifr_addr's */
761 (*have_v4 == 0 || *have_v6 == 0) && cp < cplim;
763 memmove(&u.ifreq, cp, sizeof(u.ifreq));
764 #ifdef LWRES_PLATFORM_HAVESALEN
765 #ifdef FIX_ZERO_SA_LEN
766 if (u.ifreq.ifr_addr.sa_len == 0)
767 u.ifreq.ifr_addr.sa_len = 16;
769 #ifdef HAVE_MINIMUM_IFREQ
770 cpsize = sizeof(u.ifreq);
771 if (u.ifreq.ifr_addr.sa_len > sizeof(struct sockaddr))
772 cpsize += (int)u.ifreq.ifr_addr.sa_len -
773 (int)(sizeof(struct sockaddr));
775 cpsize = sizeof(u.ifreq.ifr_name) + u.ifreq.ifr_addr.sa_len;
776 #endif /* HAVE_MINIMUM_IFREQ */
777 if (cpsize > sizeof(u.ifreq) && cpsize <= sizeof(u))
778 memmove(&u.ifreq, cp, cpsize);
779 #elif defined SIOCGIFCONF_ADDR
780 cpsize = sizeof(u.ifreq);
782 cpsize = sizeof(u.ifreq.ifr_name);
783 /* XXX maybe this should be a hard error? */
784 if (ioctl(s, SIOCGIFADDR, (char *)&u.ifreq) < 0)
787 switch (u.ifreq.ifr_addr.sa_family) {
791 &((struct sockaddr_in *)
792 &u.ifreq.ifr_addr)->sin_addr,
794 if (in4.s_addr == INADDR_ANY)
796 n = ioctl(s, SIOCGIFFLAGS, (char *)&u.ifreq);
799 if ((u.ifreq.ifr_flags & IFF_UP) == 0)
807 &((struct sockaddr_in6 *)
808 &u.ifreq.ifr_addr)->sin6_addr,
810 if (memcmp(&in6, &in6addr_any,
813 n = ioctl(s, SIOCGIFFLAGS, (char *)&u.ifreq);
816 if ((u.ifreq.ifr_flags & IFF_UP) == 0)
843 static struct hostent *
844 copyandmerge(struct hostent *he1, struct hostent *he2, int af, int *error_num)
846 struct hostent *he = NULL;
847 int addresses = 1; /* NULL terminator */
848 int names = 1; /* NULL terminator */
853 * Work out array sizes.
856 cpp = he1->h_addr_list;
857 while (*cpp != NULL) {
861 cpp = he1->h_aliases;
862 while (*cpp != NULL) {
869 cpp = he2->h_addr_list;
870 while (*cpp != NULL) {
875 cpp = he2->h_aliases;
876 while (*cpp != NULL) {
883 if (addresses == 1) {
884 *error_num = NO_ADDRESS;
888 he = malloc(sizeof(*he));
892 he->h_addr_list = malloc(sizeof(char *) * (addresses));
893 if (he->h_addr_list == NULL)
895 memset(he->h_addr_list, 0, sizeof(char *) * (addresses));
900 npp = he->h_addr_list;
902 cpp = he1->h_addr_list;
903 while (*cpp != NULL) {
904 *npp = malloc((af == AF_INET) ? INADDRSZ : IN6ADDRSZ);
908 * Convert to mapped if required.
910 if (af == AF_INET6 && he1->h_addrtype == AF_INET) {
911 memmove(*npp, in6addr_mapped,
912 sizeof(in6addr_mapped));
913 memmove(*npp + sizeof(in6addr_mapped), *cpp,
917 (af == AF_INET) ? INADDRSZ : IN6ADDRSZ);
925 cpp = he2->h_addr_list;
926 while (*cpp != NULL) {
927 *npp = malloc((af == AF_INET) ? INADDRSZ : IN6ADDRSZ);
931 * Convert to mapped if required.
933 if (af == AF_INET6 && he2->h_addrtype == AF_INET) {
934 memmove(*npp, in6addr_mapped,
935 sizeof(in6addr_mapped));
936 memmove(*npp + sizeof(in6addr_mapped), *cpp,
940 (af == AF_INET) ? INADDRSZ : IN6ADDRSZ);
947 he->h_aliases = malloc(sizeof(char *) * (names));
948 if (he->h_aliases == NULL)
950 memset(he->h_aliases, 0, sizeof(char *) * (names));
956 cpp = (he1 != NULL) ? he1->h_aliases
957 : ((he2 != NULL) ? he2->h_aliases : NULL);
958 while (cpp != NULL && *cpp != NULL) {
959 len = strlen (*cpp) + 1;
971 he->h_name = malloc(strlen((he1 != NULL) ?
972 he1->h_name : he2->h_name) + 1);
973 if (he->h_name == NULL)
975 strcpy(he->h_name, (he1 != NULL) ? he1->h_name : he2->h_name);
978 * Set address type and length.
981 he->h_length = (af == AF_INET) ? INADDRSZ : IN6ADDRSZ;
986 while (*cpp != NULL) {
993 cpp = he->h_addr_list;
994 while (*cpp != NULL) {
999 free(he->h_addr_list);
1005 *error_num = NO_RECOVERY;
1009 static struct hostent *
1010 hostfromaddr(lwres_gnbaresponse_t *addr, int af, const void *src) {
1014 he = malloc(sizeof(*he));
1017 memset(he, 0, sizeof(*he));
1020 * Set family and length.
1022 he->h_addrtype = af;
1025 he->h_length = INADDRSZ;
1028 he->h_length = IN6ADDRSZ;
1037 he->h_name = strdup(addr->realname);
1038 if (he->h_name == NULL)
1044 he->h_aliases = malloc(sizeof(char *) * (addr->naliases + 1));
1045 if (he->h_aliases == NULL)
1047 for (i = 0; i < addr->naliases; i++) {
1048 he->h_aliases[i] = strdup(addr->aliases[i]);
1049 if (he->h_aliases[i] == NULL)
1052 he->h_aliases[i] = NULL;
1057 he->h_addr_list = malloc(sizeof(char *) * 2);
1058 if (he->h_addr_list == NULL)
1060 he->h_addr_list[0] = malloc(he->h_length);
1061 if (he->h_addr_list[0] == NULL)
1063 memmove(he->h_addr_list[0], src, he->h_length);
1064 he->h_addr_list[1] = NULL;
1068 if (he != NULL && he->h_addr_list != NULL) {
1069 for (i = 0; he->h_addr_list[i] != NULL; i++)
1070 free(he->h_addr_list[i]);
1071 free(he->h_addr_list);
1073 if (he != NULL && he->h_aliases != NULL) {
1074 for (i = 0; he->h_aliases[i] != NULL; i++)
1075 free(he->h_aliases[i]);
1076 free(he->h_aliases);
1078 if (he != NULL && he->h_name != NULL)
1085 static struct hostent *
1086 hostfromname(lwres_gabnresponse_t *name, int af) {
1091 he = malloc(sizeof(*he));
1094 memset(he, 0, sizeof(*he));
1097 * Set family and length.
1099 he->h_addrtype = af;
1102 he->h_length = INADDRSZ;
1105 he->h_length = IN6ADDRSZ;
1114 he->h_name = strdup(name->realname);
1115 if (he->h_name == NULL)
1121 he->h_aliases = malloc(sizeof(char *) * (name->naliases + 1));
1122 if (he->h_aliases == NULL)
1124 for (i = 0; i < name->naliases; i++) {
1125 he->h_aliases[i] = strdup(name->aliases[i]);
1126 if (he->h_aliases[i] == NULL)
1129 he->h_aliases[i] = NULL;
1134 he->h_addr_list = malloc(sizeof(char *) * (name->naddrs + 1));
1135 if (he->h_addr_list == NULL)
1137 addr = LWRES_LIST_HEAD(name->addrs);
1139 while (addr != NULL) {
1140 he->h_addr_list[i] = malloc(he->h_length);
1141 if (he->h_addr_list[i] == NULL)
1143 memmove(he->h_addr_list[i], addr->address, he->h_length);
1144 addr = LWRES_LIST_NEXT(addr, link);
1147 he->h_addr_list[i] = NULL;
1151 if (he != NULL && he->h_addr_list != NULL) {
1152 for (i = 0; he->h_addr_list[i] != NULL; i++)
1153 free(he->h_addr_list[i]);
1154 free(he->h_addr_list);
1156 if (he != NULL && he->h_aliases != NULL) {
1157 for (i = 0; he->h_aliases[i] != NULL; i++)
1158 free(he->h_aliases[i]);
1159 free(he->h_aliases);
1161 if (he != NULL && he->h_name != NULL)