1999-06-25 22:09:35 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 1999 Internet Software Consortium.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
|
|
|
|
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
|
|
|
|
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
1999-09-02 12:07:00 +00:00
|
|
|
#include <stdio.h>
|
1999-06-25 22:09:35 +00:00
|
|
|
|
|
|
|
#include <isc/types.h>
|
|
|
|
#include <isc/assertions.h>
|
|
|
|
#include <isc/error.h>
|
|
|
|
#include <isc/sockaddr.h>
|
1999-09-02 12:07:00 +00:00
|
|
|
#include <isc/mem.h>
|
1999-06-25 22:09:35 +00:00
|
|
|
|
|
|
|
isc_boolean_t
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_equal(const isc_sockaddr_t *a, const isc_sockaddr_t *b)
|
1999-06-25 22:09:35 +00:00
|
|
|
{
|
1999-07-08 00:04:44 +00:00
|
|
|
REQUIRE(a != NULL && b != NULL);
|
1999-06-25 22:09:35 +00:00
|
|
|
|
1999-07-08 00:04:44 +00:00
|
|
|
if (a->length != b->length)
|
1999-06-25 22:09:35 +00:00
|
|
|
return (ISC_FALSE);
|
|
|
|
|
1999-07-12 18:43:53 +00:00
|
|
|
/*
|
|
|
|
* We don't just memcmp because the sin_zero field isn't always
|
|
|
|
* zero.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (a->type.sa.sa_family != b->type.sa.sa_family)
|
1999-06-25 22:09:35 +00:00
|
|
|
return (ISC_FALSE);
|
1999-07-12 18:43:53 +00:00
|
|
|
switch (a->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
if (memcmp(&a->type.sin.sin_addr, &b->type.sin.sin_addr,
|
|
|
|
sizeof a->type.sin.sin_addr) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
if (a->type.sin.sin_port != b->type.sin.sin_port)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
if (memcmp(&a->type.sin6.sin6_addr, &b->type.sin6.sin6_addr,
|
|
|
|
sizeof a->type.sin6.sin6_addr) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
if (a->type.sin6.sin6_port != b->type.sin6.sin6_port)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (memcmp(&a->type, &b->type, a->length) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
}
|
1999-07-08 00:04:44 +00:00
|
|
|
return (ISC_TRUE);
|
|
|
|
}
|
1999-06-25 22:09:35 +00:00
|
|
|
|
1999-10-25 10:07:37 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* compare just the addresses (ignore ports)
|
|
|
|
*/
|
|
|
|
isc_boolean_t
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_eqaddr(const isc_sockaddr_t *a, const isc_sockaddr_t *b)
|
1999-10-25 10:07:37 +00:00
|
|
|
{
|
|
|
|
REQUIRE(a != NULL && b != NULL);
|
|
|
|
|
|
|
|
if (a->length != b->length)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We don't just memcmp because the sin_zero field isn't always
|
|
|
|
* zero.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (a->type.sa.sa_family != b->type.sa.sa_family)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
switch (a->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
if (memcmp(&a->type.sin.sin_addr, &b->type.sin.sin_addr,
|
|
|
|
sizeof a->type.sin.sin_addr) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
if (memcmp(&a->type.sin6.sin6_addr, &b->type.sin6.sin6_addr,
|
|
|
|
sizeof a->type.sin6.sin6_addr) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (memcmp(&a->type, &b->type, a->length) != 0)
|
|
|
|
return (ISC_FALSE);
|
|
|
|
}
|
|
|
|
return (ISC_TRUE);
|
|
|
|
}
|
|
|
|
|
1999-10-29 23:46:27 +00:00
|
|
|
isc_result_t
|
|
|
|
isc_sockaddr_totext(const isc_sockaddr_t *sockaddr, isc_buffer_t *target) {
|
1999-09-02 12:07:00 +00:00
|
|
|
char abuf[sizeof "xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:255.255.255.255"];
|
1999-10-29 23:46:27 +00:00
|
|
|
unsigned int alen;
|
1999-09-02 12:07:00 +00:00
|
|
|
char pbuf[sizeof "65000"];
|
1999-10-29 23:46:27 +00:00
|
|
|
unsigned int plen;
|
|
|
|
isc_region_t avail;
|
1999-10-25 14:37:04 +00:00
|
|
|
const struct sockaddr *sa;
|
|
|
|
const struct sockaddr_in *sin;
|
|
|
|
const struct sockaddr_in6 *sin6;
|
1999-09-02 12:07:00 +00:00
|
|
|
|
|
|
|
REQUIRE(sockaddr != NULL);
|
|
|
|
|
|
|
|
sa = &sockaddr->type.sa;
|
|
|
|
switch (sa->sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
sin = &sockaddr->type.sin;
|
|
|
|
inet_ntop(sa->sa_family, &sin->sin_addr, abuf, sizeof abuf);
|
|
|
|
sprintf(pbuf, "%u", ntohs(sin->sin_port));
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
sin6 = &sockaddr->type.sin6;
|
|
|
|
inet_ntop(sa->sa_family, &sin6->sin6_addr, abuf, sizeof abuf);
|
|
|
|
sprintf(pbuf, "%u", ntohs(sin6->sin6_port));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return (NULL);
|
|
|
|
}
|
1999-10-29 23:46:27 +00:00
|
|
|
|
|
|
|
alen = strlen(abuf);
|
|
|
|
plen = strlen(pbuf);
|
|
|
|
|
|
|
|
isc_buffer_available(target, &avail);
|
|
|
|
if (alen + 1 + plen < avail.length)
|
|
|
|
return (ISC_R_NOSPACE);
|
|
|
|
|
|
|
|
isc_buffer_putmem(target, abuf, alen);
|
|
|
|
isc_buffer_putmem(target, "#", 1);
|
|
|
|
isc_buffer_putmem(target, pbuf, plen);
|
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
1999-09-02 12:07:00 +00:00
|
|
|
}
|
|
|
|
|
1999-07-08 00:04:44 +00:00
|
|
|
unsigned int
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_hash(const isc_sockaddr_t *sockaddr, isc_boolean_t address_only) {
|
1999-07-08 00:04:44 +00:00
|
|
|
unsigned int length;
|
|
|
|
const unsigned char *s;
|
|
|
|
unsigned int h = 0;
|
|
|
|
unsigned int g;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Provide a hash value for 'sockaddr'.
|
|
|
|
*/
|
1999-06-25 22:09:35 +00:00
|
|
|
|
1999-07-08 00:04:44 +00:00
|
|
|
REQUIRE(sockaddr != NULL);
|
|
|
|
|
|
|
|
if (address_only) {
|
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
return (ntohl(sockaddr->type.sin.sin_addr.s_addr));
|
|
|
|
case AF_INET6:
|
|
|
|
s = (unsigned char *)&sockaddr->type.sin6.sin6_addr;
|
|
|
|
length = sizeof sockaddr->type.sin6.sin6_addr;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
UNEXPECTED_ERROR(__FILE__, __LINE__,
|
|
|
|
"unknown address family: %d\n",
|
|
|
|
(int)sockaddr->type.sa.sa_family);
|
|
|
|
s = (unsigned char *)&sockaddr->type;
|
|
|
|
length = sockaddr->length;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
s = (unsigned char *)&sockaddr->type;
|
|
|
|
length = sockaddr->length;
|
1999-06-25 22:09:35 +00:00
|
|
|
}
|
1999-07-08 00:04:44 +00:00
|
|
|
|
|
|
|
while (length > 0) {
|
|
|
|
h = ( h << 4 ) + *s;
|
|
|
|
if ((g = ( h & 0xf0000000 )) != 0) {
|
|
|
|
h = h ^ (g >> 24);
|
|
|
|
h = h ^ g;
|
|
|
|
}
|
|
|
|
s++;
|
|
|
|
length--;
|
1999-06-25 22:09:35 +00:00
|
|
|
}
|
1999-07-08 00:04:44 +00:00
|
|
|
return (h);
|
1999-06-25 22:09:35 +00:00
|
|
|
}
|
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,
|
1999-10-02 02:54:16 +00:00
|
|
|
in_port_t port)
|
1999-07-13 01:46:15 +00:00
|
|
|
{
|
1999-07-15 20:10:38 +00:00
|
|
|
memset(sockaddr, 0, sizeof *sockaddr);
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin.sin_family = AF_INET;
|
1999-10-28 01:36:36 +00:00
|
|
|
#ifdef ISC_PLATFORM_HAVESALEN
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin.sin_len = sizeof sockaddr->type.sin;
|
|
|
|
#endif
|
|
|
|
sockaddr->type.sin.sin_addr = *ina;
|
|
|
|
sockaddr->type.sin.sin_port = htons(port);
|
|
|
|
sockaddr->length = sizeof sockaddr->type.sin;
|
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
1999-10-25 14:37:04 +00:00
|
|
|
isc_sockaddr_fromin6(isc_sockaddr_t *sockaddr, const struct in6_addr *ina6,
|
1999-10-02 02:54:16 +00:00
|
|
|
in_port_t port)
|
1999-07-13 01:46:15 +00:00
|
|
|
{
|
1999-07-15 20:10:38 +00:00
|
|
|
memset(sockaddr, 0, sizeof *sockaddr);
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin6.sin6_family = AF_INET6;
|
1999-10-28 01:36:36 +00:00
|
|
|
#ifdef ISC_PLATFORM_HAVESALEN
|
1999-07-13 01:46:15 +00:00
|
|
|
sockaddr->type.sin6.sin6_len = sizeof sockaddr->type.sin6;
|
|
|
|
#endif
|
|
|
|
sockaddr->type.sin6.sin6_addr = *ina6;
|
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
|
|
|
sockaddr->length = sizeof sockaddr->type.sin6;
|
|
|
|
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,
|
1999-10-02 02:54:16 +00:00
|
|
|
in_port_t port)
|
1999-07-15 20:10:38 +00:00
|
|
|
{
|
|
|
|
memset(sockaddr, 0, sizeof *sockaddr);
|
|
|
|
sockaddr->type.sin6.sin6_family = AF_INET6;
|
1999-10-28 01:36:36 +00:00
|
|
|
#ifdef ISC_PLATFORM_HAVESALEN
|
1999-07-15 20:10:38 +00:00
|
|
|
sockaddr->type.sin6.sin6_len = sizeof sockaddr->type.sin6;
|
|
|
|
#endif
|
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;
|
1999-07-15 20:10:38 +00:00
|
|
|
memcpy(&sockaddr->type.sin6.sin6_addr.s6_addr[12], ina, 4);
|
|
|
|
sockaddr->type.sin6.sin6_port = htons(port);
|
|
|
|
sockaddr->length = sizeof sockaddr->type.sin6;
|
|
|
|
ISC_LINK_INIT(sockaddr, link);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
1999-10-25 14:37:04 +00: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.
|
|
|
|
*/
|
|
|
|
return (sockaddr->type.sa.sa_family);
|
|
|
|
#else
|
|
|
|
switch (sockaddr->type.sa.sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
return (PF_INET);
|
|
|
|
case AF_INET6:
|
|
|
|
return (PF_INET);
|
|
|
|
default:
|
|
|
|
FATAL_ERROR(__FILE__, __LINE__, "unknown address family");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|