1999-06-25 22:09:35 +00:00
|
|
|
/*
|
2018-02-23 09:53:12 +01:00
|
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
2000-08-01 01:33:37 +00:00
|
|
|
*
|
2021-06-03 08:37:05 +02:00
|
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
|
|
*
|
2016-06-27 14:56:38 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
2020-09-14 16:20:40 -07:00
|
|
|
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
2018-02-23 09:53:12 +01:00
|
|
|
*
|
|
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
|
|
* information regarding copyright ownership.
|
1999-06-25 22:09:35 +00:00
|
|
|
*/
|
|
|
|
|
2005-04-27 04:57:32 +00:00
|
|
|
/*! \file */
|
2000-06-22 22:00:42 +00:00
|
|
|
|
2022-12-16 11:04:39 +00:00
|
|
|
#include <netdb.h>
|
2018-04-17 08:29:14 -07:00
|
|
|
#include <stdbool.h>
|
1999-09-02 12:07:00 +00:00
|
|
|
#include <stdio.h>
|
1999-06-25 22:09:35 +00:00
|
|
|
|
2000-04-27 23:29:28 +00:00
|
|
|
#include <isc/buffer.h>
|
2003-07-25 02:22:26 +00:00
|
|
|
#include <isc/hash.h>
|
2000-02-09 22:52:37 +00:00
|
|
|
#include <isc/netaddr.h>
|
2000-04-10 21:57:24 +00:00
|
|
|
#include <isc/region.h>
|
|
|
|
#include <isc/sockaddr.h>
|
2000-05-08 14:38:29 +00:00
|
|
|
#include <isc/string.h>
|
2000-04-28 01:12:23 +00:00
|
|
|
#include <isc/util.h>
|
1999-06-25 22:09:35 +00:00
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_equal(const isc_sockaddr_t *a, const isc_sockaddr_t *b) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_sockaddr_compare(a, b,
|
|
|
|
ISC_SOCKADDR_CMPADDR |
|
|
|
|
ISC_SOCKADDR_CMPPORT |
|
|
|
|
ISC_SOCKADDR_CMPSCOPE);
|
2004-11-22 23:29:10 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_eqaddr(const isc_sockaddr_t *a, const isc_sockaddr_t *b) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_sockaddr_compare(
|
|
|
|
a, b, ISC_SOCKADDR_CMPADDR | ISC_SOCKADDR_CMPSCOPE);
|
2004-11-22 23:29:10 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2004-11-22 23:29:10 +00:00
|
|
|
isc_sockaddr_compare(const isc_sockaddr_t *a, const isc_sockaddr_t *b,
|
2020-02-13 14:44:37 -08:00
|
|
|
unsigned int flags) {
|
1999-07-08 00:04:44 +00:00
|
|
|
REQUIRE(a != NULL && b != NULL);
|
1999-06-25 22:09:35 +00:00
|
|
|
|
2020-02-13 21:48:23 +01:00
|
|
|
if (a->length != b->length) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
1999-06-25 22:09:35 +00:00
|
|
|
|
1999-07-12 18:43:53 +00:00
|
|
|
/*
|
|
|
|
* We don't just memcmp because the sin_zero field isn't always
|
|
|
|
* zero.
|
|
|
|
*/
|
|
|
|
|
2020-02-13 21:48:23 +01:00
|
|
|
if (a->type.sa.sa_family != b->type.sa.sa_family) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
1999-07-12 18:43:53 +00:00
|
|
|
switch (a->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
2004-11-22 23:29:10 +00:00
|
|
|
if ((flags & ISC_SOCKADDR_CMPADDR) != 0 &&
|
|
|
|
memcmp(&a->type.sin.sin_addr, &b->type.sin.sin_addr,
|
2020-02-13 14:44:37 -08:00
|
|
|
sizeof(a->type.sin.sin_addr)) != 0)
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2004-11-22 23:29:10 +00:00
|
|
|
if ((flags & ISC_SOCKADDR_CMPPORT) != 0 &&
|
2020-02-13 14:44:37 -08:00
|
|
|
a->type.sin.sin_port != b->type.sin.sin_port)
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
1999-07-12 18:43:53 +00:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2004-11-22 23:29:10 +00:00
|
|
|
if ((flags & ISC_SOCKADDR_CMPADDR) != 0 &&
|
|
|
|
memcmp(&a->type.sin6.sin6_addr, &b->type.sin6.sin6_addr,
|
2020-02-13 14:44:37 -08:00
|
|
|
sizeof(a->type.sin6.sin6_addr)) != 0)
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2004-11-22 23:29:10 +00:00
|
|
|
/*
|
|
|
|
* If ISC_SOCKADDR_CMPSCOPEZERO is set then don't return
|
2018-04-17 08:29:14 -07:00
|
|
|
* false if one of the scopes in zero.
|
2004-11-22 23:29:10 +00:00
|
|
|
*/
|
|
|
|
if ((flags & ISC_SOCKADDR_CMPSCOPE) != 0 &&
|
|
|
|
a->type.sin6.sin6_scope_id != b->type.sin6.sin6_scope_id &&
|
|
|
|
((flags & ISC_SOCKADDR_CMPSCOPEZERO) == 0 ||
|
2020-02-12 13:59:18 +01:00
|
|
|
(a->type.sin6.sin6_scope_id != 0 &&
|
2020-02-13 14:44:37 -08:00
|
|
|
b->type.sin6.sin6_scope_id != 0)))
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2004-11-22 23:29:10 +00:00
|
|
|
if ((flags & ISC_SOCKADDR_CMPPORT) != 0 &&
|
2020-02-13 14:44:37 -08:00
|
|
|
a->type.sin6.sin6_port != b->type.sin6.sin6_port)
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
1999-10-25 10:07:37 +00:00
|
|
|
break;
|
|
|
|
default:
|
2020-02-13 21:48:23 +01:00
|
|
|
if (memcmp(&a->type, &b->type, a->length) != 0) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
1999-10-25 10:07:37 +00:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return true;
|
1999-10-25 10:07:37 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
1999-11-29 20:00:19 +00:00
|
|
|
isc_sockaddr_eqaddrprefix(const isc_sockaddr_t *a, const isc_sockaddr_t *b,
|
2020-02-13 14:44:37 -08:00
|
|
|
unsigned int prefixlen) {
|
2000-02-09 22:52:37 +00:00
|
|
|
isc_netaddr_t na, nb;
|
|
|
|
isc_netaddr_fromsockaddr(&na, a);
|
|
|
|
isc_netaddr_fromsockaddr(&nb, b);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_eqprefix(&na, &nb, prefixlen);
|
1999-11-29 20:00:19 +00:00
|
|
|
}
|
|
|
|
|
1999-10-29 23:46:27 +00:00
|
|
|
isc_result_t
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_totext(const isc_sockaddr_t *sockaddr, isc_buffer_t *target) {
|
|
|
|
isc_result_t result;
|
2000-05-26 00:41:57 +00:00
|
|
|
isc_netaddr_t netaddr;
|
2020-02-13 14:44:37 -08:00
|
|
|
char pbuf[sizeof("65000")];
|
|
|
|
unsigned int plen;
|
|
|
|
isc_region_t avail;
|
1999-09-02 12:07:00 +00:00
|
|
|
|
|
|
|
REQUIRE(sockaddr != NULL);
|
|
|
|
|
2000-06-02 17:48:31 +00:00
|
|
|
/*
|
|
|
|
* Do the port first, giving us the opportunity to check for
|
2000-08-01 01:33:37 +00:00
|
|
|
* unsupported address families before calling
|
2000-06-02 17:48:31 +00:00
|
|
|
* isc_netaddr_fromsockaddr().
|
|
|
|
*/
|
2000-05-26 00:41:57 +00:00
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
1999-09-02 12:07:00 +00:00
|
|
|
case AF_INET:
|
2020-02-12 13:59:18 +01:00
|
|
|
snprintf(pbuf, sizeof(pbuf), "%u",
|
|
|
|
ntohs(sockaddr->type.sin.sin_port));
|
1999-09-02 12:07:00 +00:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2020-02-12 13:59:18 +01:00
|
|
|
snprintf(pbuf, sizeof(pbuf), "%u",
|
|
|
|
ntohs(sockaddr->type.sin6.sin6_port));
|
1999-09-02 12:07:00 +00:00
|
|
|
break;
|
|
|
|
default:
|
2024-11-19 10:38:03 +01:00
|
|
|
return ISC_R_FAILURE;
|
1999-09-02 12:07:00 +00:00
|
|
|
}
|
2000-08-01 01:33:37 +00:00
|
|
|
|
1999-10-29 23:46:27 +00:00
|
|
|
plen = strlen(pbuf);
|
2000-05-26 00:41:57 +00:00
|
|
|
INSIST(plen < sizeof(pbuf));
|
2000-08-01 01:33:37 +00:00
|
|
|
|
2000-06-02 17:48:31 +00:00
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
|
|
|
result = isc_netaddr_totext(&netaddr, target);
|
2020-02-13 21:48:23 +01:00
|
|
|
if (result != ISC_R_SUCCESS) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return result;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2000-08-01 01:33:37 +00:00
|
|
|
|
2020-02-13 21:48:23 +01:00
|
|
|
if (1 + plen + 1 > isc_buffer_availablelength(target)) {
|
2024-11-19 10:38:03 +01:00
|
|
|
return ISC_R_NOSPACE;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2000-08-01 01:33:37 +00:00
|
|
|
|
2000-06-01 17:20:56 +00:00
|
|
|
isc_buffer_putmem(target, (const unsigned char *)"#", 1);
|
|
|
|
isc_buffer_putmem(target, (const unsigned char *)pbuf, plen);
|
1999-10-29 23:46:27 +00:00
|
|
|
|
2000-06-01 17:20:56 +00:00
|
|
|
/*
|
|
|
|
* Null terminate after used region.
|
|
|
|
*/
|
103. [func] libisc buffer API changes for <isc/buffer.h>:
Added:
isc_buffer_base(b) (pointer)
isc_buffer_current(b) (pointer)
isc_buffer_active(b) (pointer)
isc_buffer_used(b) (pointer)
isc_buffer_length(b) (int)
isc_buffer_usedlength(b) (int)
isc_buffer_consumedlength(b) (int)
isc_buffer_remaininglength(b) (int)
isc_buffer_activelength(b) (int)
isc_buffer_availablelength(b) (int)
Removed:
ISC_BUFFER_USEDCOUNT(b)
ISC_BUFFER_AVAILABLECOUNT(b)
isc_buffer_type(b)
Changed names:
isc_buffer_used(b, r) ->
isc_buffer_usedregion(b, r)
isc_buffer_available(b, r) ->
isc_buffer_available_region(b, r)
isc_buffer_consumed(b, r) ->
isc_buffer_consumedregion(b, r)
isc_buffer_active(b, r) ->
isc_buffer_activeregion(b, r)
isc_buffer_remaining(b, r) ->
isc_buffer_remainingregion(b, r)
Buffer types were removed, so the ISC_BUFFERTYPE_*
macros are no more, and the type argument to
isc_buffer_init and isc_buffer_allocate were removed.
isc_buffer_putstr is now void (instead of isc_result_t)
and requires that the caller ensure that there
is enough available buffer space for the string.
2000-04-27 00:03:12 +00:00
|
|
|
isc_buffer_availableregion(target, &avail);
|
1999-10-29 23:53:26 +00:00
|
|
|
INSIST(avail.length >= 1);
|
|
|
|
avail.base[0] = '\0';
|
|
|
|
|
2024-11-19 10:38:03 +01:00
|
|
|
return ISC_R_SUCCESS;
|
1999-09-02 12:07:00 +00:00
|
|
|
}
|
|
|
|
|
2000-05-14 20:42:43 +00:00
|
|
|
void
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_format(const isc_sockaddr_t *sa, char *array, unsigned int size) {
|
2000-05-14 20:42:43 +00:00
|
|
|
isc_result_t result;
|
|
|
|
isc_buffer_t buf;
|
|
|
|
|
2020-02-13 21:48:23 +01:00
|
|
|
if (size == 0U) {
|
2011-02-21 06:18:03 +00:00
|
|
|
return;
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2011-02-21 06:18:03 +00:00
|
|
|
|
2000-05-14 20:42:43 +00:00
|
|
|
isc_buffer_init(&buf, array, size);
|
|
|
|
result = isc_sockaddr_totext(sa, &buf);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
2000-12-06 00:30:32 +00:00
|
|
|
/*
|
|
|
|
* The message is the same as in netaddr.c.
|
|
|
|
*/
|
2020-02-12 13:59:18 +01:00
|
|
|
snprintf(array, size, "<unknown address, family %u>",
|
2000-05-14 20:42:43 +00:00
|
|
|
sa->type.sa.sa_family);
|
|
|
|
array[size - 1] = '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-18 09:59:10 +02:00
|
|
|
void
|
|
|
|
isc_sockaddr_hash_ex(isc_hash32_t *hash, const isc_sockaddr_t *sockaddr,
|
|
|
|
bool address_only) {
|
2023-09-12 16:33:19 +02:00
|
|
|
REQUIRE(sockaddr != NULL);
|
|
|
|
|
|
|
|
size_t len = 0;
|
|
|
|
const uint8_t *s = NULL;
|
2020-02-13 14:44:37 -08:00
|
|
|
unsigned int p = 0;
|
2001-09-17 06:58:18 +00:00
|
|
|
const struct in6_addr *in6;
|
1999-07-08 00:04:44 +00:00
|
|
|
|
2001-10-02 06:06:15 +00:00
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
2023-09-12 16:33:19 +02:00
|
|
|
s = (const uint8_t *)&sockaddr->type.sin.sin_addr;
|
|
|
|
len = sizeof(sockaddr->type.sin.sin_addr.s_addr);
|
|
|
|
if (!address_only) {
|
|
|
|
p = ntohs(sockaddr->type.sin.sin_port);
|
|
|
|
}
|
2001-10-02 06:06:15 +00:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
in6 = &sockaddr->type.sin6.sin6_addr;
|
2023-09-12 16:33:19 +02:00
|
|
|
s = (const uint8_t *)in6;
|
2001-10-02 06:06:15 +00:00
|
|
|
if (IN6_IS_ADDR_V4MAPPED(in6)) {
|
2012-10-23 22:04:06 -07:00
|
|
|
s += 12;
|
2023-09-12 16:33:19 +02:00
|
|
|
len = sizeof(sockaddr->type.sin.sin_addr.s_addr);
|
2020-02-13 21:48:23 +01:00
|
|
|
} else {
|
2023-09-12 16:33:19 +02:00
|
|
|
len = sizeof(sockaddr->type.sin6.sin6_addr);
|
|
|
|
}
|
|
|
|
if (!address_only) {
|
|
|
|
p = ntohs(sockaddr->type.sin6.sin6_port);
|
2020-02-13 21:48:23 +01:00
|
|
|
}
|
2001-10-02 06:06:15 +00:00
|
|
|
break;
|
|
|
|
default:
|
2023-09-12 16:33:19 +02:00
|
|
|
UNREACHABLE();
|
1999-06-25 22:09:35 +00:00
|
|
|
}
|
2003-07-25 02:22:26 +00:00
|
|
|
|
2023-09-18 09:59:10 +02:00
|
|
|
isc_hash32_hash(hash, s, len, true);
|
2019-04-04 13:51:09 +02:00
|
|
|
if (!address_only) {
|
2023-09-18 09:59:10 +02:00
|
|
|
isc_hash32_hash(hash, &p, sizeof(p), true);
|
2019-04-04 13:51:09 +02:00
|
|
|
}
|
2023-09-18 09:59:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t
|
|
|
|
isc_sockaddr_hash(const isc_sockaddr_t *sockaddr, bool address_only) {
|
|
|
|
isc_hash32_t hash;
|
|
|
|
|
|
|
|
isc_hash32_init(&hash);
|
|
|
|
|
|
|
|
isc_sockaddr_hash_ex(&hash, sockaddr, address_only);
|
2003-07-25 02:22:26 +00:00
|
|
|
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_hash32_finalize(&hash);
|
1999-06-25 22:09:35 +00:00
|
|
|
}
|
1999-07-13 01:46:15 +00:00
|
|
|
|
2000-05-06 01:49:43 +00:00
|
|
|
void
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_any(isc_sockaddr_t *sockaddr) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
2000-05-06 01:49:43 +00:00
|
|
|
sockaddr->type.sin.sin_family = AF_INET;
|
|
|
|
sockaddr->type.sin.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
sockaddr->type.sin.sin_port = 0;
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin);
|
2000-05-06 01:49:43 +00:00
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_any6(isc_sockaddr_t *sockaddr) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
2000-05-06 01:49:43 +00:00
|
|
|
sockaddr->type.sin6.sin6_family = AF_INET6;
|
|
|
|
sockaddr->type.sin6.sin6_addr = in6addr_any;
|
|
|
|
sockaddr->type.sin6.sin6_port = 0;
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin6);
|
2000-05-06 01:49:43 +00:00
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
1999-07-13 01:46:15 +00:00
|
|
|
void
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_fromin(isc_sockaddr_t *sockaddr, const struct in_addr *ina,
|
2020-02-13 14:44:37 -08:00
|
|
|
in_port_t port) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin.sin_family = AF_INET;
|
|
|
|
sockaddr->type.sin.sin_addr = *ina;
|
|
|
|
sockaddr->type.sin.sin_port = htons(port);
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin);
|
1999-07-13 01:46:15 +00:00
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
2000-08-09 18:57:16 +00:00
|
|
|
void
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_anyofpf(isc_sockaddr_t *sockaddr, int pf) {
|
2020-02-12 13:59:18 +01:00
|
|
|
switch (pf) {
|
|
|
|
case AF_INET:
|
|
|
|
isc_sockaddr_any(sockaddr);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
isc_sockaddr_any6(sockaddr);
|
|
|
|
break;
|
|
|
|
default:
|
2021-10-11 12:50:17 +02:00
|
|
|
UNREACHABLE();
|
2020-02-12 13:59:18 +01:00
|
|
|
}
|
2000-08-09 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
1999-07-13 01:46:15 +00:00
|
|
|
void
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_fromin6(isc_sockaddr_t *sockaddr, const struct in6_addr *ina6,
|
2020-02-13 14:44:37 -08:00
|
|
|
in_port_t port) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin6.sin6_family = AF_INET6;
|
|
|
|
sockaddr->type.sin6.sin6_addr = *ina6;
|
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin6);
|
1999-07-13 01:46:15 +00:00
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
1999-07-15 20:10:38 +00:00
|
|
|
|
|
|
|
void
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_v6fromin(isc_sockaddr_t *sockaddr, const struct in_addr *ina,
|
2020-02-13 14:44:37 -08:00
|
|
|
in_port_t port) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
1999-07-15 20:10:38 +00:00
|
|
|
sockaddr->type.sin6.sin6_family = AF_INET6;
|
1999-08-28 05:33:29 +00:00
|
|
|
sockaddr->type.sin6.sin6_addr.s6_addr[10] = 0xff;
|
|
|
|
sockaddr->type.sin6.sin6_addr.s6_addr[11] = 0xff;
|
2014-01-08 16:27:10 -08:00
|
|
|
memmove(&sockaddr->type.sin6.sin6_addr.s6_addr[12], ina, 4);
|
1999-07-15 20:10:38 +00:00
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin6);
|
1999-07-15 20:10:38 +00:00
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_pf(const isc_sockaddr_t *sockaddr) {
|
1999-07-15 20:10:38 +00:00
|
|
|
/*
|
|
|
|
* Get the protocol family of 'sockaddr'.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if (AF_INET == PF_INET && AF_INET6 == PF_INET6)
|
|
|
|
/*
|
|
|
|
* Assume that PF_xxx == AF_xxx for all AF and PF.
|
|
|
|
*/
|
2024-11-19 10:38:03 +01:00
|
|
|
return sockaddr->type.sa.sa_family;
|
2020-02-13 21:48:23 +01:00
|
|
|
#else /* if (AF_INET == PF_INET && AF_INET6 == PF_INET6) */
|
1999-07-15 20:10:38 +00:00
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
2024-11-19 10:38:03 +01:00
|
|
|
return PF_INET;
|
1999-07-15 20:10:38 +00:00
|
|
|
case AF_INET6:
|
2024-11-19 10:38:03 +01:00
|
|
|
return PF_INET6;
|
1999-07-15 20:10:38 +00:00
|
|
|
default:
|
2022-10-14 16:07:07 +01:00
|
|
|
FATAL_ERROR("unknown address family: %d",
|
2000-12-06 00:30:32 +00:00
|
|
|
(int)sockaddr->type.sa.sa_family);
|
1999-07-15 20:10:38 +00:00
|
|
|
}
|
2020-02-13 21:48:23 +01:00
|
|
|
#endif /* if (AF_INET == PF_INET && AF_INET6 == PF_INET6) */
|
1999-07-15 20:10:38 +00:00
|
|
|
}
|
2000-01-26 19:22:41 +00:00
|
|
|
|
2000-02-09 22:52:37 +00:00
|
|
|
void
|
|
|
|
isc_sockaddr_fromnetaddr(isc_sockaddr_t *sockaddr, const isc_netaddr_t *na,
|
2020-02-13 14:44:37 -08:00
|
|
|
in_port_t port) {
|
2001-11-27 01:56:32 +00:00
|
|
|
memset(sockaddr, 0, sizeof(*sockaddr));
|
2000-02-09 22:52:37 +00:00
|
|
|
sockaddr->type.sin.sin_family = na->family;
|
|
|
|
switch (na->family) {
|
|
|
|
case AF_INET:
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin);
|
2000-02-09 22:52:37 +00:00
|
|
|
sockaddr->type.sin.sin_addr = na->type.in;
|
|
|
|
sockaddr->type.sin.sin_port = htons(port);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2001-11-27 01:56:32 +00:00
|
|
|
sockaddr->length = sizeof(sockaddr->type.sin6);
|
2014-01-08 16:27:10 -08:00
|
|
|
memmove(&sockaddr->type.sin6.sin6_addr, &na->type.in6, 16);
|
2002-10-24 03:52:35 +00:00
|
|
|
sockaddr->type.sin6.sin6_scope_id = isc_netaddr_getzone(na);
|
2000-02-09 22:52:37 +00:00
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
|
|
|
break;
|
2010-11-17 23:47:09 +00:00
|
|
|
default:
|
2021-10-11 12:50:17 +02:00
|
|
|
UNREACHABLE();
|
2000-02-09 22:52:37 +00:00
|
|
|
}
|
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
2000-01-26 19:22:41 +00:00
|
|
|
void
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_setport(isc_sockaddr_t *sockaddr, in_port_t port) {
|
2000-01-26 19:22:41 +00:00
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
sockaddr->type.sin.sin_port = htons(port);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
|
|
|
break;
|
|
|
|
default:
|
2022-10-14 16:07:07 +01:00
|
|
|
FATAL_ERROR("unknown address family: %d",
|
2000-12-06 00:30:32 +00:00
|
|
|
(int)sockaddr->type.sa.sa_family);
|
2000-01-26 19:22:41 +00:00
|
|
|
}
|
|
|
|
}
|
2000-01-31 21:52:17 +00:00
|
|
|
|
|
|
|
in_port_t
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_getport(const isc_sockaddr_t *sockaddr) {
|
2000-01-31 21:52:17 +00:00
|
|
|
in_port_t port = 0;
|
|
|
|
|
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
port = ntohs(sockaddr->type.sin.sin_port);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
port = ntohs(sockaddr->type.sin6.sin6_port);
|
|
|
|
break;
|
|
|
|
default:
|
2022-10-14 16:07:07 +01:00
|
|
|
FATAL_ERROR("unknown address family: %d",
|
2000-12-06 00:30:32 +00:00
|
|
|
(int)sockaddr->type.sa.sa_family);
|
2000-01-31 21:52:17 +00:00
|
|
|
}
|
|
|
|
|
2024-11-19 10:38:03 +01:00
|
|
|
return port;
|
2000-01-31 21:52:17 +00:00
|
|
|
}
|
2000-06-15 18:23:54 +00:00
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_ismulticast(const isc_sockaddr_t *sockaddr) {
|
2000-06-15 18:23:54 +00:00
|
|
|
isc_netaddr_t netaddr;
|
|
|
|
|
2005-02-23 01:09:23 +00:00
|
|
|
if (sockaddr->type.sa.sa_family == AF_INET ||
|
2020-02-13 14:44:37 -08:00
|
|
|
sockaddr->type.sa.sa_family == AF_INET6)
|
|
|
|
{
|
2005-02-23 01:09:23 +00:00
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_ismulticast(&netaddr);
|
2005-02-23 01:09:23 +00:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2000-06-15 18:23:54 +00:00
|
|
|
}
|
2002-04-03 06:38:38 +00:00
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_isexperimental(const isc_sockaddr_t *sockaddr) {
|
2004-02-20 00:52:46 +00:00
|
|
|
isc_netaddr_t netaddr;
|
|
|
|
|
|
|
|
if (sockaddr->type.sa.sa_family == AF_INET) {
|
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_isexperimental(&netaddr);
|
2004-02-20 00:52:46 +00:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2004-02-20 00:52:46 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_issitelocal(const isc_sockaddr_t *sockaddr) {
|
2002-04-03 06:38:38 +00:00
|
|
|
isc_netaddr_t netaddr;
|
|
|
|
|
|
|
|
if (sockaddr->type.sa.sa_family == AF_INET6) {
|
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_issitelocal(&netaddr);
|
2002-04-03 06:38:38 +00:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2002-04-03 06:38:38 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_islinklocal(const isc_sockaddr_t *sockaddr) {
|
2002-04-03 06:38:38 +00:00
|
|
|
isc_netaddr_t netaddr;
|
|
|
|
|
|
|
|
if (sockaddr->type.sa.sa_family == AF_INET6) {
|
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_islinklocal(&netaddr);
|
2002-04-03 06:38:38 +00:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2002-04-03 06:38:38 +00:00
|
|
|
}
|
2005-02-23 01:09:23 +00:00
|
|
|
|
2018-04-17 08:29:14 -07:00
|
|
|
bool
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_isnetzero(const isc_sockaddr_t *sockaddr) {
|
2015-10-16 08:00:15 +11:00
|
|
|
isc_netaddr_t netaddr;
|
|
|
|
|
|
|
|
if (sockaddr->type.sa.sa_family == AF_INET) {
|
|
|
|
isc_netaddr_fromsockaddr(&netaddr, sockaddr);
|
2024-11-19 10:38:03 +01:00
|
|
|
return isc_netaddr_isnetzero(&netaddr);
|
2015-10-16 08:00:15 +11:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2015-10-16 08:00:15 +11:00
|
|
|
}
|
|
|
|
|
2019-01-15 11:01:13 +01:00
|
|
|
isc_result_t
|
2020-02-13 14:44:37 -08:00
|
|
|
isc_sockaddr_fromsockaddr(isc_sockaddr_t *isa, const struct sockaddr *sa) {
|
2019-01-15 11:01:13 +01:00
|
|
|
unsigned int length = 0;
|
|
|
|
|
|
|
|
switch (sa->sa_family) {
|
2020-02-12 13:59:18 +01:00
|
|
|
case AF_INET:
|
|
|
|
length = sizeof(isa->type.sin);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
length = sizeof(isa->type.sin6);
|
|
|
|
break;
|
|
|
|
default:
|
2024-11-19 10:38:03 +01:00
|
|
|
return ISC_R_NOTIMPLEMENTED;
|
2019-01-15 11:01:13 +01:00
|
|
|
}
|
|
|
|
|
2023-01-31 13:30:12 -08:00
|
|
|
*isa = (isc_sockaddr_t){ .length = length,
|
|
|
|
.link = ISC_LINK_INITIALIZER };
|
2020-05-13 17:37:51 +02:00
|
|
|
memmove(isa, sa, length);
|
2019-01-15 11:01:13 +01:00
|
|
|
|
2024-11-19 10:38:03 +01:00
|
|
|
return ISC_R_SUCCESS;
|
2019-01-15 11:01:13 +01:00
|
|
|
}
|
2024-02-13 11:42:44 +11:00
|
|
|
|
|
|
|
bool
|
|
|
|
isc_sockaddr_disabled(const isc_sockaddr_t *sockaddr) {
|
|
|
|
if ((sockaddr->type.sa.sa_family == AF_INET &&
|
|
|
|
isc_net_probeipv4() == ISC_R_DISABLED) ||
|
|
|
|
(sockaddr->type.sa.sa_family == AF_INET6 &&
|
|
|
|
isc_net_probeipv6() == ISC_R_DISABLED))
|
|
|
|
{
|
2024-11-19 10:38:03 +01:00
|
|
|
return true;
|
2024-02-13 11:42:44 +11:00
|
|
|
}
|
2024-11-19 10:38:03 +01:00
|
|
|
return false;
|
2024-02-13 11:42:44 +11:00
|
|
|
}
|