1*00b67f09SDavid van Moolenbroek /* $NetBSD: gethost.c,v 1.7 2015/07/08 17:29:00 christos Exp $ */
2*00b67f09SDavid van Moolenbroek
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek * Copyright (C) 2004, 2005, 2007, 2013-2015 Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek * Copyright (C) 2000, 2001 Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek *
7*00b67f09SDavid van Moolenbroek * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek *
11*00b67f09SDavid van Moolenbroek * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek */
19*00b67f09SDavid van Moolenbroek
20*00b67f09SDavid van Moolenbroek /* Id: gethost.c,v 1.34 2007/06/19 23:47:22 tbox Exp */
21*00b67f09SDavid van Moolenbroek
22*00b67f09SDavid van Moolenbroek /*! \file */
23*00b67f09SDavid van Moolenbroek
24*00b67f09SDavid van Moolenbroek /**
25*00b67f09SDavid van Moolenbroek * These functions provide hostname-to-address and address-to-hostname
26*00b67f09SDavid van Moolenbroek * lookups by means of the lightweight resolver. They are similar to the
27*00b67f09SDavid van Moolenbroek * standard gethostent(3) functions provided by most operating systems.
28*00b67f09SDavid van Moolenbroek * They use a struct hostent which is usually defined in <namedb.h>.
29*00b67f09SDavid van Moolenbroek *
30*00b67f09SDavid van Moolenbroek * \code
31*00b67f09SDavid van Moolenbroek * struct hostent {
32*00b67f09SDavid van Moolenbroek * char *h_name; // official name of host
33*00b67f09SDavid van Moolenbroek * char **h_aliases; // alias list
34*00b67f09SDavid van Moolenbroek * int h_addrtype; // host address type
35*00b67f09SDavid van Moolenbroek * int h_length; // length of address
36*00b67f09SDavid van Moolenbroek * char **h_addr_list; // list of addresses from name server
37*00b67f09SDavid van Moolenbroek * };
38*00b67f09SDavid van Moolenbroek * #define h_addr h_addr_list[0] // address, for backward compatibility
39*00b67f09SDavid van Moolenbroek * \endcode
40*00b67f09SDavid van Moolenbroek *
41*00b67f09SDavid van Moolenbroek * The members of this structure are:
42*00b67f09SDavid van Moolenbroek *
43*00b67f09SDavid van Moolenbroek * \li h_name:
44*00b67f09SDavid van Moolenbroek * The official (canonical) name of the host.
45*00b67f09SDavid van Moolenbroek *
46*00b67f09SDavid van Moolenbroek * \li h_aliases:
47*00b67f09SDavid van Moolenbroek * A NULL-terminated array of alternate names (nicknames) for the
48*00b67f09SDavid van Moolenbroek * host.
49*00b67f09SDavid van Moolenbroek *
50*00b67f09SDavid van Moolenbroek * \li h_addrtype:
51*00b67f09SDavid van Moolenbroek * The type of address being returned -- PF_INET or PF_INET6.
52*00b67f09SDavid van Moolenbroek *
53*00b67f09SDavid van Moolenbroek * \li h_length:
54*00b67f09SDavid van Moolenbroek * The length of the address in bytes.
55*00b67f09SDavid van Moolenbroek *
56*00b67f09SDavid van Moolenbroek * \li h_addr_list:
57*00b67f09SDavid van Moolenbroek * A NULL terminated array of network addresses for the host. Host
58*00b67f09SDavid van Moolenbroek * addresses are returned in network byte order.
59*00b67f09SDavid van Moolenbroek *
60*00b67f09SDavid van Moolenbroek * For backward compatibility with very old software, h_addr is the first
61*00b67f09SDavid van Moolenbroek * address in h_addr_list.
62*00b67f09SDavid van Moolenbroek *
63*00b67f09SDavid van Moolenbroek * lwres_gethostent(), lwres_sethostent(), lwres_endhostent(),
64*00b67f09SDavid van Moolenbroek * lwres_gethostent_r(), lwres_sethostent_r() and lwres_endhostent_r()
65*00b67f09SDavid van Moolenbroek * provide iteration over the known host entries on systems that provide
66*00b67f09SDavid van Moolenbroek * such functionality through facilities like /etc/hosts or NIS. The
67*00b67f09SDavid van Moolenbroek * lightweight resolver does not currently implement these functions; it
68*00b67f09SDavid van Moolenbroek * only provides them as stub functions that always return failure.
69*00b67f09SDavid van Moolenbroek *
70*00b67f09SDavid van Moolenbroek * lwres_gethostbyname() and lwres_gethostbyname2() look up the hostname
71*00b67f09SDavid van Moolenbroek * name. lwres_gethostbyname() always looks for an IPv4 address while
72*00b67f09SDavid van Moolenbroek * lwres_gethostbyname2() looks for an address of protocol family af:
73*00b67f09SDavid van Moolenbroek * either PF_INET or PF_INET6 -- IPv4 or IPV6 addresses respectively.
74*00b67f09SDavid van Moolenbroek * Successful calls of the functions return a struct hostent for the name
75*00b67f09SDavid van Moolenbroek * that was looked up. NULL is returned if the lookups by
76*00b67f09SDavid van Moolenbroek * lwres_gethostbyname() or lwres_gethostbyname2() fail.
77*00b67f09SDavid van Moolenbroek *
78*00b67f09SDavid van Moolenbroek * Reverse lookups of addresses are performed by lwres_gethostbyaddr().
79*00b67f09SDavid van Moolenbroek * addr is an address of length len bytes and protocol family type --
80*00b67f09SDavid van Moolenbroek * PF_INET or PF_INET6. lwres_gethostbyname_r() is a thread-safe function
81*00b67f09SDavid van Moolenbroek * for forward lookups. If an error occurs, an error code is returned in
82*00b67f09SDavid van Moolenbroek * *error. resbuf is a pointer to a struct hostent which is initialised
83*00b67f09SDavid van Moolenbroek * by a successful call to lwres_gethostbyname_r() . buf is a buffer of
84*00b67f09SDavid van Moolenbroek * length len bytes which is used to store the h_name, h_aliases, and
85*00b67f09SDavid van Moolenbroek * h_addr_list elements of the struct hostent returned in resbuf.
86*00b67f09SDavid van Moolenbroek * Successful calls to lwres_gethostbyname_r() return resbuf, which is a
87*00b67f09SDavid van Moolenbroek * pointer to the struct hostent it created.
88*00b67f09SDavid van Moolenbroek *
89*00b67f09SDavid van Moolenbroek * lwres_gethostbyaddr_r() is a thread-safe function that performs a
90*00b67f09SDavid van Moolenbroek * reverse lookup of address addr which is len bytes long and is of
91*00b67f09SDavid van Moolenbroek * protocol family type -- PF_INET or PF_INET6. If an error occurs, the
92*00b67f09SDavid van Moolenbroek * error code is returned in *error. The other function parameters are
93*00b67f09SDavid van Moolenbroek * identical to those in lwres_gethostbyname_r(). resbuf is a pointer to
94*00b67f09SDavid van Moolenbroek * a struct hostent which is initialised by a successful call to
95*00b67f09SDavid van Moolenbroek * lwres_gethostbyaddr_r(). buf is a buffer of length len bytes which is
96*00b67f09SDavid van Moolenbroek * used to store the h_name, h_aliases, and h_addr_list elements of the
97*00b67f09SDavid van Moolenbroek * struct hostent returned in resbuf. Successful calls to
98*00b67f09SDavid van Moolenbroek * lwres_gethostbyaddr_r() return resbuf, which is a pointer to the
99*00b67f09SDavid van Moolenbroek * struct hostent it created.
100*00b67f09SDavid van Moolenbroek *
101*00b67f09SDavid van Moolenbroek * \section gethost_return Return Values
102*00b67f09SDavid van Moolenbroek *
103*00b67f09SDavid van Moolenbroek * The functions lwres_gethostbyname(), lwres_gethostbyname2(),
104*00b67f09SDavid van Moolenbroek * lwres_gethostbyaddr(), and lwres_gethostent() return NULL to indicate
105*00b67f09SDavid van Moolenbroek * an error. In this case the global variable lwres_h_errno will contain
106*00b67f09SDavid van Moolenbroek * one of the following error codes defined in \link netdb.h <lwres/netdb.h>:\endlink
107*00b67f09SDavid van Moolenbroek *
108*00b67f09SDavid van Moolenbroek * \li #HOST_NOT_FOUND:
109*00b67f09SDavid van Moolenbroek * The host or address was not found.
110*00b67f09SDavid van Moolenbroek *
111*00b67f09SDavid van Moolenbroek * \li #TRY_AGAIN:
112*00b67f09SDavid van Moolenbroek * A recoverable error occurred, e.g., a timeout. Retrying the
113*00b67f09SDavid van Moolenbroek * lookup may succeed.
114*00b67f09SDavid van Moolenbroek *
115*00b67f09SDavid van Moolenbroek * \li #NO_RECOVERY:
116*00b67f09SDavid van Moolenbroek * A non-recoverable error occurred.
117*00b67f09SDavid van Moolenbroek *
118*00b67f09SDavid van Moolenbroek * \li #NO_DATA:
119*00b67f09SDavid van Moolenbroek * The name exists, but has no address information associated with
120*00b67f09SDavid van Moolenbroek * it (or vice versa in the case of a reverse lookup). The code
121*00b67f09SDavid van Moolenbroek * NO_ADDRESS is accepted as a synonym for NO_DATA for backwards
122*00b67f09SDavid van Moolenbroek * compatibility.
123*00b67f09SDavid van Moolenbroek *
124*00b67f09SDavid van Moolenbroek * lwres_hstrerror() translates these error codes to suitable error
125*00b67f09SDavid van Moolenbroek * messages.
126*00b67f09SDavid van Moolenbroek *
127*00b67f09SDavid van Moolenbroek * lwres_gethostent() and lwres_gethostent_r() always return NULL.
128*00b67f09SDavid van Moolenbroek *
129*00b67f09SDavid van Moolenbroek * Successful calls to lwres_gethostbyname_r() and
130*00b67f09SDavid van Moolenbroek * lwres_gethostbyaddr_r() return resbuf, a pointer to the struct hostent
131*00b67f09SDavid van Moolenbroek * that was initialised by these functions. They return NULL if the
132*00b67f09SDavid van Moolenbroek * lookups fail or if buf was too small to hold the list of addresses and
133*00b67f09SDavid van Moolenbroek * names referenced by the h_name, h_aliases, and h_addr_list elements of
134*00b67f09SDavid van Moolenbroek * the struct hostent. If buf was too small, both lwres_gethostbyname_r()
135*00b67f09SDavid van Moolenbroek * and lwres_gethostbyaddr_r() set the global variable errno to ERANGE.
136*00b67f09SDavid van Moolenbroek *
137*00b67f09SDavid van Moolenbroek * \section gethost_see See Also
138*00b67f09SDavid van Moolenbroek *
139*00b67f09SDavid van Moolenbroek * gethostent(), \link getipnode.c getipnode\endlink, lwres_hstrerror()
140*00b67f09SDavid van Moolenbroek *
141*00b67f09SDavid van Moolenbroek * \section gethost_bugs Bugs
142*00b67f09SDavid van Moolenbroek *
143*00b67f09SDavid van Moolenbroek * lwres_gethostbyname(), lwres_gethostbyname2(), lwres_gethostbyaddr()
144*00b67f09SDavid van Moolenbroek * and lwres_endhostent() are not thread safe; they return pointers to
145*00b67f09SDavid van Moolenbroek * static data and provide error codes through a global variable.
146*00b67f09SDavid van Moolenbroek * Thread-safe versions for name and address lookup are provided by
147*00b67f09SDavid van Moolenbroek * lwres_gethostbyname_r(), and lwres_gethostbyaddr_r() respectively.
148*00b67f09SDavid van Moolenbroek *
149*00b67f09SDavid van Moolenbroek * The resolver daemon does not currently support any non-DNS name
150*00b67f09SDavid van Moolenbroek * services such as /etc/hosts or NIS, consequently the above functions
151*00b67f09SDavid van Moolenbroek * don't, either.
152*00b67f09SDavid van Moolenbroek */
153*00b67f09SDavid van Moolenbroek
154*00b67f09SDavid van Moolenbroek #include <config.h>
155*00b67f09SDavid van Moolenbroek
156*00b67f09SDavid van Moolenbroek #include <errno.h>
157*00b67f09SDavid van Moolenbroek #include <string.h>
158*00b67f09SDavid van Moolenbroek #ifdef HAVE_INTTYPES_H
159*00b67f09SDavid van Moolenbroek #include <inttypes.h> /* uintptr_t */
160*00b67f09SDavid van Moolenbroek #endif
161*00b67f09SDavid van Moolenbroek
162*00b67f09SDavid van Moolenbroek #include <lwres/net.h>
163*00b67f09SDavid van Moolenbroek #include <lwres/netdb.h>
164*00b67f09SDavid van Moolenbroek
165*00b67f09SDavid van Moolenbroek #include "assert_p.h"
166*00b67f09SDavid van Moolenbroek
167*00b67f09SDavid van Moolenbroek #define LWRES_ALIGNBYTES (sizeof(char *) - 1)
168*00b67f09SDavid van Moolenbroek #define LWRES_ALIGN(p) \
169*00b67f09SDavid van Moolenbroek (((uintptr_t)(p) + LWRES_ALIGNBYTES) &~ LWRES_ALIGNBYTES)
170*00b67f09SDavid van Moolenbroek
171*00b67f09SDavid van Moolenbroek static struct hostent *he = NULL;
172*00b67f09SDavid van Moolenbroek static int copytobuf(struct hostent *, struct hostent *, char *, int);
173*00b67f09SDavid van Moolenbroek
174*00b67f09SDavid van Moolenbroek /*% Always looks for an IPv4 address. */
175*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostbyname(const char * name)176*00b67f09SDavid van Moolenbroek lwres_gethostbyname(const char *name) {
177*00b67f09SDavid van Moolenbroek
178*00b67f09SDavid van Moolenbroek if (he != NULL)
179*00b67f09SDavid van Moolenbroek lwres_freehostent(he);
180*00b67f09SDavid van Moolenbroek
181*00b67f09SDavid van Moolenbroek he = lwres_getipnodebyname(name, AF_INET, 0, &lwres_h_errno);
182*00b67f09SDavid van Moolenbroek return (he);
183*00b67f09SDavid van Moolenbroek }
184*00b67f09SDavid van Moolenbroek
185*00b67f09SDavid van Moolenbroek /*% Looks for either an IPv4 or IPv6 address. */
186*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostbyname2(const char * name,int af)187*00b67f09SDavid van Moolenbroek lwres_gethostbyname2(const char *name, int af) {
188*00b67f09SDavid van Moolenbroek if (he != NULL)
189*00b67f09SDavid van Moolenbroek lwres_freehostent(he);
190*00b67f09SDavid van Moolenbroek
191*00b67f09SDavid van Moolenbroek he = lwres_getipnodebyname(name, af, 0, &lwres_h_errno);
192*00b67f09SDavid van Moolenbroek return (he);
193*00b67f09SDavid van Moolenbroek }
194*00b67f09SDavid van Moolenbroek
195*00b67f09SDavid van Moolenbroek /*% Reverse lookup of addresses. */
196*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostbyaddr(const char * addr,int len,int type)197*00b67f09SDavid van Moolenbroek lwres_gethostbyaddr(const char *addr, int len, int type) {
198*00b67f09SDavid van Moolenbroek
199*00b67f09SDavid van Moolenbroek if (he != NULL)
200*00b67f09SDavid van Moolenbroek lwres_freehostent(he);
201*00b67f09SDavid van Moolenbroek
202*00b67f09SDavid van Moolenbroek he = lwres_getipnodebyaddr(addr, len, type, &lwres_h_errno);
203*00b67f09SDavid van Moolenbroek return (he);
204*00b67f09SDavid van Moolenbroek }
205*00b67f09SDavid van Moolenbroek
206*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
207*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostent(void)208*00b67f09SDavid van Moolenbroek lwres_gethostent(void) {
209*00b67f09SDavid van Moolenbroek if (he != NULL)
210*00b67f09SDavid van Moolenbroek lwres_freehostent(he);
211*00b67f09SDavid van Moolenbroek
212*00b67f09SDavid van Moolenbroek return (NULL);
213*00b67f09SDavid van Moolenbroek }
214*00b67f09SDavid van Moolenbroek
215*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
216*00b67f09SDavid van Moolenbroek void
lwres_sethostent(int stayopen)217*00b67f09SDavid van Moolenbroek lwres_sethostent(int stayopen) {
218*00b67f09SDavid van Moolenbroek /*
219*00b67f09SDavid van Moolenbroek * Empty.
220*00b67f09SDavid van Moolenbroek */
221*00b67f09SDavid van Moolenbroek UNUSED(stayopen);
222*00b67f09SDavid van Moolenbroek }
223*00b67f09SDavid van Moolenbroek
224*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
225*00b67f09SDavid van Moolenbroek void
lwres_endhostent(void)226*00b67f09SDavid van Moolenbroek lwres_endhostent(void) {
227*00b67f09SDavid van Moolenbroek /*
228*00b67f09SDavid van Moolenbroek * Empty.
229*00b67f09SDavid van Moolenbroek */
230*00b67f09SDavid van Moolenbroek }
231*00b67f09SDavid van Moolenbroek
232*00b67f09SDavid van Moolenbroek /*% Thread-safe function for forward lookups. */
233*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostbyname_r(const char * name,struct hostent * resbuf,char * buf,int buflen,int * error)234*00b67f09SDavid van Moolenbroek lwres_gethostbyname_r(const char *name, struct hostent *resbuf,
235*00b67f09SDavid van Moolenbroek char *buf, int buflen, int *error)
236*00b67f09SDavid van Moolenbroek {
237*00b67f09SDavid van Moolenbroek struct hostent *myhe;
238*00b67f09SDavid van Moolenbroek int res;
239*00b67f09SDavid van Moolenbroek
240*00b67f09SDavid van Moolenbroek myhe = lwres_getipnodebyname(name, AF_INET, 0, error);
241*00b67f09SDavid van Moolenbroek if (myhe == NULL)
242*00b67f09SDavid van Moolenbroek return (NULL);
243*00b67f09SDavid van Moolenbroek res = copytobuf(myhe, resbuf, buf, buflen);
244*00b67f09SDavid van Moolenbroek lwres_freehostent(myhe);
245*00b67f09SDavid van Moolenbroek if (res != 0) {
246*00b67f09SDavid van Moolenbroek errno = ERANGE;
247*00b67f09SDavid van Moolenbroek return (NULL);
248*00b67f09SDavid van Moolenbroek }
249*00b67f09SDavid van Moolenbroek return (resbuf);
250*00b67f09SDavid van Moolenbroek }
251*00b67f09SDavid van Moolenbroek
252*00b67f09SDavid van Moolenbroek /*% Thread-safe reverse lookup. */
253*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostbyaddr_r(const char * addr,int len,int type,struct hostent * resbuf,char * buf,int buflen,int * error)254*00b67f09SDavid van Moolenbroek lwres_gethostbyaddr_r(const char *addr, int len, int type,
255*00b67f09SDavid van Moolenbroek struct hostent *resbuf, char *buf, int buflen,
256*00b67f09SDavid van Moolenbroek int *error)
257*00b67f09SDavid van Moolenbroek {
258*00b67f09SDavid van Moolenbroek struct hostent *myhe;
259*00b67f09SDavid van Moolenbroek int res;
260*00b67f09SDavid van Moolenbroek
261*00b67f09SDavid van Moolenbroek myhe = lwres_getipnodebyaddr(addr, len, type, error);
262*00b67f09SDavid van Moolenbroek if (myhe == NULL)
263*00b67f09SDavid van Moolenbroek return (NULL);
264*00b67f09SDavid van Moolenbroek res = copytobuf(myhe, resbuf, buf, buflen);
265*00b67f09SDavid van Moolenbroek lwres_freehostent(myhe);
266*00b67f09SDavid van Moolenbroek if (res != 0) {
267*00b67f09SDavid van Moolenbroek errno = ERANGE;
268*00b67f09SDavid van Moolenbroek return (NULL);
269*00b67f09SDavid van Moolenbroek }
270*00b67f09SDavid van Moolenbroek return (resbuf);
271*00b67f09SDavid van Moolenbroek }
272*00b67f09SDavid van Moolenbroek
273*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
274*00b67f09SDavid van Moolenbroek struct hostent *
lwres_gethostent_r(struct hostent * resbuf,char * buf,int buflen,int * error)275*00b67f09SDavid van Moolenbroek lwres_gethostent_r(struct hostent *resbuf, char *buf, int buflen, int *error) {
276*00b67f09SDavid van Moolenbroek UNUSED(resbuf);
277*00b67f09SDavid van Moolenbroek UNUSED(buf);
278*00b67f09SDavid van Moolenbroek UNUSED(buflen);
279*00b67f09SDavid van Moolenbroek *error = 0;
280*00b67f09SDavid van Moolenbroek return (NULL);
281*00b67f09SDavid van Moolenbroek }
282*00b67f09SDavid van Moolenbroek
283*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
284*00b67f09SDavid van Moolenbroek void
lwres_sethostent_r(int stayopen)285*00b67f09SDavid van Moolenbroek lwres_sethostent_r(int stayopen) {
286*00b67f09SDavid van Moolenbroek /*
287*00b67f09SDavid van Moolenbroek * Empty.
288*00b67f09SDavid van Moolenbroek */
289*00b67f09SDavid van Moolenbroek UNUSED(stayopen);
290*00b67f09SDavid van Moolenbroek }
291*00b67f09SDavid van Moolenbroek
292*00b67f09SDavid van Moolenbroek /*% Stub function. Always returns failure. */
293*00b67f09SDavid van Moolenbroek void
lwres_endhostent_r(void)294*00b67f09SDavid van Moolenbroek lwres_endhostent_r(void) {
295*00b67f09SDavid van Moolenbroek /*
296*00b67f09SDavid van Moolenbroek * Empty.
297*00b67f09SDavid van Moolenbroek */
298*00b67f09SDavid van Moolenbroek }
299*00b67f09SDavid van Moolenbroek
300*00b67f09SDavid van Moolenbroek static int
copytobuf(struct hostent * src,struct hostent * hptr,char * buf,int buflen)301*00b67f09SDavid van Moolenbroek copytobuf(struct hostent *src, struct hostent *hptr, char *buf, int buflen) {
302*00b67f09SDavid van Moolenbroek char *cp;
303*00b67f09SDavid van Moolenbroek char **ptr;
304*00b67f09SDavid van Moolenbroek int i, n;
305*00b67f09SDavid van Moolenbroek int nptr, len;
306*00b67f09SDavid van Moolenbroek
307*00b67f09SDavid van Moolenbroek /*
308*00b67f09SDavid van Moolenbroek * Find out the amount of space required to store the answer.
309*00b67f09SDavid van Moolenbroek */
310*00b67f09SDavid van Moolenbroek nptr = 2; /* NULL ptrs */
311*00b67f09SDavid van Moolenbroek len = (int)((char *)LWRES_ALIGN(buf) - buf);
312*00b67f09SDavid van Moolenbroek for (i = 0; src->h_addr_list[i]; i++, nptr++) {
313*00b67f09SDavid van Moolenbroek len += src->h_length;
314*00b67f09SDavid van Moolenbroek }
315*00b67f09SDavid van Moolenbroek for (i = 0; src->h_aliases[i]; i++, nptr++) {
316*00b67f09SDavid van Moolenbroek len += strlen(src->h_aliases[i]) + 1;
317*00b67f09SDavid van Moolenbroek }
318*00b67f09SDavid van Moolenbroek len += strlen(src->h_name) + 1;
319*00b67f09SDavid van Moolenbroek len += nptr * sizeof(char*);
320*00b67f09SDavid van Moolenbroek
321*00b67f09SDavid van Moolenbroek if (len > buflen) {
322*00b67f09SDavid van Moolenbroek return (-1);
323*00b67f09SDavid van Moolenbroek }
324*00b67f09SDavid van Moolenbroek
325*00b67f09SDavid van Moolenbroek /*
326*00b67f09SDavid van Moolenbroek * Copy address size and type.
327*00b67f09SDavid van Moolenbroek */
328*00b67f09SDavid van Moolenbroek hptr->h_addrtype = src->h_addrtype;
329*00b67f09SDavid van Moolenbroek n = hptr->h_length = src->h_length;
330*00b67f09SDavid van Moolenbroek
331*00b67f09SDavid van Moolenbroek ptr = (char **)LWRES_ALIGN(buf);
332*00b67f09SDavid van Moolenbroek cp = (char *)LWRES_ALIGN(buf) + nptr * sizeof(char *);
333*00b67f09SDavid van Moolenbroek
334*00b67f09SDavid van Moolenbroek /*
335*00b67f09SDavid van Moolenbroek * Copy address list.
336*00b67f09SDavid van Moolenbroek */
337*00b67f09SDavid van Moolenbroek hptr->h_addr_list = ptr;
338*00b67f09SDavid van Moolenbroek for (i = 0; src->h_addr_list[i]; i++, ptr++) {
339*00b67f09SDavid van Moolenbroek memmove(cp, src->h_addr_list[i], n);
340*00b67f09SDavid van Moolenbroek hptr->h_addr_list[i] = cp;
341*00b67f09SDavid van Moolenbroek cp += n;
342*00b67f09SDavid van Moolenbroek }
343*00b67f09SDavid van Moolenbroek hptr->h_addr_list[i] = NULL;
344*00b67f09SDavid van Moolenbroek ptr++;
345*00b67f09SDavid van Moolenbroek
346*00b67f09SDavid van Moolenbroek /*
347*00b67f09SDavid van Moolenbroek * Copy official name.
348*00b67f09SDavid van Moolenbroek */
349*00b67f09SDavid van Moolenbroek n = strlen(src->h_name) + 1;
350*00b67f09SDavid van Moolenbroek strcpy(cp, src->h_name);
351*00b67f09SDavid van Moolenbroek hptr->h_name = cp;
352*00b67f09SDavid van Moolenbroek cp += n;
353*00b67f09SDavid van Moolenbroek
354*00b67f09SDavid van Moolenbroek /*
355*00b67f09SDavid van Moolenbroek * Copy aliases.
356*00b67f09SDavid van Moolenbroek */
357*00b67f09SDavid van Moolenbroek hptr->h_aliases = ptr;
358*00b67f09SDavid van Moolenbroek for (i = 0; src->h_aliases[i]; i++) {
359*00b67f09SDavid van Moolenbroek n = strlen(src->h_aliases[i]) + 1;
360*00b67f09SDavid van Moolenbroek strcpy(cp, src->h_aliases[i]);
361*00b67f09SDavid van Moolenbroek hptr->h_aliases[i] = cp;
362*00b67f09SDavid van Moolenbroek cp += n;
363*00b67f09SDavid van Moolenbroek }
364*00b67f09SDavid van Moolenbroek hptr->h_aliases[i] = NULL;
365*00b67f09SDavid van Moolenbroek
366*00b67f09SDavid van Moolenbroek return (0);
367*00b67f09SDavid van Moolenbroek }
368