2 * Copyright (C) 2004-2007, 2010-2012, 2014, 2015 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.
26 #include <isc/buffer.h>
29 #include <isc/netaddr.h>
30 #include <isc/print.h>
31 #include <isc/region.h>
32 #include <isc/sockaddr.h>
33 #include <isc/string.h>
37 isc_sockaddr_equal(const isc_sockaddr_t *a, const isc_sockaddr_t *b) {
38 return (isc_sockaddr_compare(a, b, ISC_SOCKADDR_CMPADDR|
40 ISC_SOCKADDR_CMPSCOPE));
44 isc_sockaddr_eqaddr(const isc_sockaddr_t *a, const isc_sockaddr_t *b) {
45 return (isc_sockaddr_compare(a, b, ISC_SOCKADDR_CMPADDR|
46 ISC_SOCKADDR_CMPSCOPE));
50 isc_sockaddr_compare(const isc_sockaddr_t *a, const isc_sockaddr_t *b,
53 REQUIRE(a != NULL && b != NULL);
55 if (a->length != b->length)
59 * We don't just memcmp because the sin_zero field isn't always
63 if (a->type.sa.sa_family != b->type.sa.sa_family)
65 switch (a->type.sa.sa_family) {
67 if ((flags & ISC_SOCKADDR_CMPADDR) != 0 &&
68 memcmp(&a->type.sin.sin_addr, &b->type.sin.sin_addr,
69 sizeof(a->type.sin.sin_addr)) != 0)
71 if ((flags & ISC_SOCKADDR_CMPPORT) != 0 &&
72 a->type.sin.sin_port != b->type.sin.sin_port)
76 if ((flags & ISC_SOCKADDR_CMPADDR) != 0 &&
77 memcmp(&a->type.sin6.sin6_addr, &b->type.sin6.sin6_addr,
78 sizeof(a->type.sin6.sin6_addr)) != 0)
80 #ifdef ISC_PLATFORM_HAVESCOPEID
82 * If ISC_SOCKADDR_CMPSCOPEZERO is set then don't return
83 * ISC_FALSE if one of the scopes in zero.
85 if ((flags & ISC_SOCKADDR_CMPSCOPE) != 0 &&
86 a->type.sin6.sin6_scope_id != b->type.sin6.sin6_scope_id &&
87 ((flags & ISC_SOCKADDR_CMPSCOPEZERO) == 0 ||
88 (a->type.sin6.sin6_scope_id != 0 &&
89 b->type.sin6.sin6_scope_id != 0)))
92 if ((flags & ISC_SOCKADDR_CMPPORT) != 0 &&
93 a->type.sin6.sin6_port != b->type.sin6.sin6_port)
97 if (memcmp(&a->type, &b->type, a->length) != 0)
104 isc_sockaddr_eqaddrprefix(const isc_sockaddr_t *a, const isc_sockaddr_t *b,
105 unsigned int prefixlen)
107 isc_netaddr_t na, nb;
108 isc_netaddr_fromsockaddr(&na, a);
109 isc_netaddr_fromsockaddr(&nb, b);
110 return (isc_netaddr_eqprefix(&na, &nb, prefixlen));
114 isc_sockaddr_totext(const isc_sockaddr_t *sockaddr, isc_buffer_t *target) {
116 isc_netaddr_t netaddr;
117 char pbuf[sizeof("65000")];
121 REQUIRE(sockaddr != NULL);
124 * Do the port first, giving us the opportunity to check for
125 * unsupported address families before calling
126 * isc_netaddr_fromsockaddr().
128 switch (sockaddr->type.sa.sa_family) {
130 snprintf(pbuf, sizeof(pbuf), "%u", ntohs(sockaddr->type.sin.sin_port));
133 snprintf(pbuf, sizeof(pbuf), "%u", ntohs(sockaddr->type.sin6.sin6_port));
135 #ifdef ISC_PLAFORM_HAVESYSUNH
137 plen = strlen(sockaddr->type.sunix.sun_path);
138 if (plen >= isc_buffer_availablelength(target))
139 return (ISC_R_NOSPACE);
141 isc_buffer_putmem(target, sockaddr->type.sunix.sun_path, plen);
144 * Null terminate after used region.
146 isc_buffer_availableregion(target, &avail);
147 INSIST(avail.length >= 1);
148 avail.base[0] = '\0';
150 return (ISC_R_SUCCESS);
153 return (ISC_R_FAILURE);
157 INSIST(plen < sizeof(pbuf));
159 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
160 result = isc_netaddr_totext(&netaddr, target);
161 if (result != ISC_R_SUCCESS)
164 if (1 + plen + 1 > isc_buffer_availablelength(target))
165 return (ISC_R_NOSPACE);
167 isc_buffer_putmem(target, (const unsigned char *)"#", 1);
168 isc_buffer_putmem(target, (const unsigned char *)pbuf, plen);
171 * Null terminate after used region.
173 isc_buffer_availableregion(target, &avail);
174 INSIST(avail.length >= 1);
175 avail.base[0] = '\0';
177 return (ISC_R_SUCCESS);
181 isc_sockaddr_format(const isc_sockaddr_t *sa, char *array, unsigned int size) {
188 isc_buffer_init(&buf, array, size);
189 result = isc_sockaddr_totext(sa, &buf);
190 if (result != ISC_R_SUCCESS) {
192 * The message is the same as in netaddr.c.
194 snprintf(array, size,
195 isc_msgcat_get(isc_msgcat, ISC_MSGSET_NETADDR,
197 "<unknown address, family %u>"),
198 sa->type.sa.sa_family);
199 array[size - 1] = '\0';
204 isc_sockaddr_hash(const isc_sockaddr_t *sockaddr, isc_boolean_t address_only) {
205 unsigned int length = 0;
206 const unsigned char *s = NULL;
209 const struct in6_addr *in6;
211 REQUIRE(sockaddr != NULL);
213 switch (sockaddr->type.sa.sa_family) {
215 s = (const unsigned char *)&sockaddr->type.sin.sin_addr;
216 p = ntohs(sockaddr->type.sin.sin_port);
217 length = sizeof(sockaddr->type.sin.sin_addr.s_addr);
220 in6 = &sockaddr->type.sin6.sin6_addr;
221 s = (const unsigned char *)in6;
222 if (IN6_IS_ADDR_V4MAPPED(in6)) {
224 length = sizeof(sockaddr->type.sin.sin_addr.s_addr);
226 length = sizeof(sockaddr->type.sin6.sin6_addr);
227 p = ntohs(sockaddr->type.sin6.sin6_port);
230 UNEXPECTED_ERROR(__FILE__, __LINE__,
231 isc_msgcat_get(isc_msgcat,
233 ISC_MSG_UNKNOWNFAMILY,
234 "unknown address family: %d"),
235 (int)sockaddr->type.sa.sa_family);
236 s = (const unsigned char *)&sockaddr->type;
237 length = sockaddr->length;
241 h = isc_hash_function(s, length, ISC_TRUE, NULL);
243 h = isc_hash_function(&p, sizeof(p), ISC_TRUE, &h);
249 isc_sockaddr_any(isc_sockaddr_t *sockaddr)
251 memset(sockaddr, 0, sizeof(*sockaddr));
252 sockaddr->type.sin.sin_family = AF_INET;
253 #ifdef ISC_PLATFORM_HAVESALEN
254 sockaddr->type.sin.sin_len = sizeof(sockaddr->type.sin);
256 sockaddr->type.sin.sin_addr.s_addr = INADDR_ANY;
257 sockaddr->type.sin.sin_port = 0;
258 sockaddr->length = sizeof(sockaddr->type.sin);
259 ISC_LINK_INIT(sockaddr, link);
263 isc_sockaddr_any6(isc_sockaddr_t *sockaddr)
265 memset(sockaddr, 0, sizeof(*sockaddr));
266 sockaddr->type.sin6.sin6_family = AF_INET6;
267 #ifdef ISC_PLATFORM_HAVESALEN
268 sockaddr->type.sin6.sin6_len = sizeof(sockaddr->type.sin6);
270 sockaddr->type.sin6.sin6_addr = in6addr_any;
271 sockaddr->type.sin6.sin6_port = 0;
272 sockaddr->length = sizeof(sockaddr->type.sin6);
273 ISC_LINK_INIT(sockaddr, link);
277 isc_sockaddr_fromin(isc_sockaddr_t *sockaddr, const struct in_addr *ina,
280 memset(sockaddr, 0, sizeof(*sockaddr));
281 sockaddr->type.sin.sin_family = AF_INET;
282 #ifdef ISC_PLATFORM_HAVESALEN
283 sockaddr->type.sin.sin_len = sizeof(sockaddr->type.sin);
285 sockaddr->type.sin.sin_addr = *ina;
286 sockaddr->type.sin.sin_port = htons(port);
287 sockaddr->length = sizeof(sockaddr->type.sin);
288 ISC_LINK_INIT(sockaddr, link);
292 isc_sockaddr_anyofpf(isc_sockaddr_t *sockaddr, int pf) {
295 isc_sockaddr_any(sockaddr);
298 isc_sockaddr_any6(sockaddr);
306 isc_sockaddr_fromin6(isc_sockaddr_t *sockaddr, const struct in6_addr *ina6,
309 memset(sockaddr, 0, sizeof(*sockaddr));
310 sockaddr->type.sin6.sin6_family = AF_INET6;
311 #ifdef ISC_PLATFORM_HAVESALEN
312 sockaddr->type.sin6.sin6_len = sizeof(sockaddr->type.sin6);
314 sockaddr->type.sin6.sin6_addr = *ina6;
315 sockaddr->type.sin6.sin6_port = htons(port);
316 sockaddr->length = sizeof(sockaddr->type.sin6);
317 ISC_LINK_INIT(sockaddr, link);
321 isc_sockaddr_v6fromin(isc_sockaddr_t *sockaddr, const struct in_addr *ina,
324 memset(sockaddr, 0, sizeof(*sockaddr));
325 sockaddr->type.sin6.sin6_family = AF_INET6;
326 #ifdef ISC_PLATFORM_HAVESALEN
327 sockaddr->type.sin6.sin6_len = sizeof(sockaddr->type.sin6);
329 sockaddr->type.sin6.sin6_addr.s6_addr[10] = 0xff;
330 sockaddr->type.sin6.sin6_addr.s6_addr[11] = 0xff;
331 memmove(&sockaddr->type.sin6.sin6_addr.s6_addr[12], ina, 4);
332 sockaddr->type.sin6.sin6_port = htons(port);
333 sockaddr->length = sizeof(sockaddr->type.sin6);
334 ISC_LINK_INIT(sockaddr, link);
338 isc_sockaddr_pf(const isc_sockaddr_t *sockaddr) {
341 * Get the protocol family of 'sockaddr'.
344 #if (AF_INET == PF_INET && AF_INET6 == PF_INET6)
346 * Assume that PF_xxx == AF_xxx for all AF and PF.
348 return (sockaddr->type.sa.sa_family);
350 switch (sockaddr->type.sa.sa_family) {
356 FATAL_ERROR(__FILE__, __LINE__,
357 isc_msgcat_get(isc_msgcat, ISC_MSGSET_SOCKADDR,
358 ISC_MSG_UNKNOWNFAMILY,
359 "unknown address family: %d"),
360 (int)sockaddr->type.sa.sa_family);
366 isc_sockaddr_fromnetaddr(isc_sockaddr_t *sockaddr, const isc_netaddr_t *na,
369 memset(sockaddr, 0, sizeof(*sockaddr));
370 sockaddr->type.sin.sin_family = na->family;
371 switch (na->family) {
373 sockaddr->length = sizeof(sockaddr->type.sin);
374 #ifdef ISC_PLATFORM_HAVESALEN
375 sockaddr->type.sin.sin_len = sizeof(sockaddr->type.sin);
377 sockaddr->type.sin.sin_addr = na->type.in;
378 sockaddr->type.sin.sin_port = htons(port);
381 sockaddr->length = sizeof(sockaddr->type.sin6);
382 #ifdef ISC_PLATFORM_HAVESALEN
383 sockaddr->type.sin6.sin6_len = sizeof(sockaddr->type.sin6);
385 memmove(&sockaddr->type.sin6.sin6_addr, &na->type.in6, 16);
386 #ifdef ISC_PLATFORM_HAVESCOPEID
387 sockaddr->type.sin6.sin6_scope_id = isc_netaddr_getzone(na);
389 sockaddr->type.sin6.sin6_port = htons(port);
394 ISC_LINK_INIT(sockaddr, link);
398 isc_sockaddr_setport(isc_sockaddr_t *sockaddr, in_port_t port) {
399 switch (sockaddr->type.sa.sa_family) {
401 sockaddr->type.sin.sin_port = htons(port);
404 sockaddr->type.sin6.sin6_port = htons(port);
407 FATAL_ERROR(__FILE__, __LINE__,
408 isc_msgcat_get(isc_msgcat, ISC_MSGSET_SOCKADDR,
409 ISC_MSG_UNKNOWNFAMILY,
410 "unknown address family: %d"),
411 (int)sockaddr->type.sa.sa_family);
416 isc_sockaddr_getport(const isc_sockaddr_t *sockaddr) {
419 switch (sockaddr->type.sa.sa_family) {
421 port = ntohs(sockaddr->type.sin.sin_port);
424 port = ntohs(sockaddr->type.sin6.sin6_port);
427 FATAL_ERROR(__FILE__, __LINE__,
428 isc_msgcat_get(isc_msgcat, ISC_MSGSET_SOCKADDR,
429 ISC_MSG_UNKNOWNFAMILY,
430 "unknown address family: %d"),
431 (int)sockaddr->type.sa.sa_family);
438 isc_sockaddr_ismulticast(const isc_sockaddr_t *sockaddr) {
439 isc_netaddr_t netaddr;
441 if (sockaddr->type.sa.sa_family == AF_INET ||
442 sockaddr->type.sa.sa_family == AF_INET6) {
443 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
444 return (isc_netaddr_ismulticast(&netaddr));
450 isc_sockaddr_isexperimental(const isc_sockaddr_t *sockaddr) {
451 isc_netaddr_t netaddr;
453 if (sockaddr->type.sa.sa_family == AF_INET) {
454 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
455 return (isc_netaddr_isexperimental(&netaddr));
461 isc_sockaddr_issitelocal(const isc_sockaddr_t *sockaddr) {
462 isc_netaddr_t netaddr;
464 if (sockaddr->type.sa.sa_family == AF_INET6) {
465 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
466 return (isc_netaddr_issitelocal(&netaddr));
472 isc_sockaddr_islinklocal(const isc_sockaddr_t *sockaddr) {
473 isc_netaddr_t netaddr;
475 if (sockaddr->type.sa.sa_family == AF_INET6) {
476 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
477 return (isc_netaddr_islinklocal(&netaddr));
483 isc_sockaddr_isnetzero(const isc_sockaddr_t *sockaddr) {
484 isc_netaddr_t netaddr;
486 if (sockaddr->type.sa.sa_family == AF_INET) {
487 isc_netaddr_fromsockaddr(&netaddr, sockaddr);
488 return (isc_netaddr_isnetzero(&netaddr));
494 isc_sockaddr_frompath(isc_sockaddr_t *sockaddr, const char *path) {
495 #ifdef ISC_PLATFORM_HAVESYSUNH
496 if (strlen(path) >= sizeof(sockaddr->type.sunix.sun_path))
497 return (ISC_R_NOSPACE);
498 memset(sockaddr, 0, sizeof(*sockaddr));
499 sockaddr->length = sizeof(sockaddr->type.sunix);
500 sockaddr->type.sunix.sun_family = AF_UNIX;
501 #ifdef ISC_PLATFORM_HAVESALEN
502 sockaddr->type.sunix.sun_len =
503 (unsigned char)sizeof(sockaddr->type.sunix);
505 strcpy(sockaddr->type.sunix.sun_path, path);
506 return (ISC_R_SUCCESS);
510 return (ISC_R_NOTIMPLEMENTED);