/*
* Copyright (C) Igor Sysoev
* Copyright (C) NGINX, Inc.
*/
#include <nxt_main.h>
#if (NXT_INET6)
static u_char *nxt_inet6_ntop(u_char *addr, u_char *buf, u_char *end);
#endif
static nxt_sockaddr_t *nxt_sockaddr_unix_parse(nxt_mp_t *mp, nxt_str_t *addr);
static nxt_sockaddr_t *nxt_sockaddr_inet6_parse(nxt_mp_t *mp, nxt_str_t *addr);
static nxt_sockaddr_t *nxt_sockaddr_inet_parse(nxt_mp_t *mp, nxt_str_t *addr);
nxt_sockaddr_t *
nxt_sockaddr_cache_alloc(nxt_event_engine_t *engine, nxt_listen_socket_t *ls)
{
size_t size;
uint8_t hint;
nxt_sockaddr_t *sa;
hint = NXT_EVENT_ENGINE_NO_MEM_HINT;
size = offsetof(nxt_sockaddr_t, u) + ls->socklen + ls->address_length;
sa = nxt_event_engine_mem_alloc(engine, &hint, size);
if (nxt_fast_path(sa != NULL)) {
/* Zero only beginning of structure up to sockaddr_un.sun_path[1]. */
nxt_memzero(sa, offsetof(nxt_sockaddr_t, u.sockaddr.sa_data[1]));
sa->cache_hint = hint;
sa->socklen = ls->socklen;
sa->length = ls->address_length;
sa->type = ls->sockaddr->type;
/*
* Set address family for unspecified Unix domain socket,
* because these sockaddr's are not updated by old BSD systems,
* see comment in nxt_conn_io_accept().
*/
sa->u.sockaddr.sa_family = ls->sockaddr->u.sockaddr.sa_family;
}
return sa;
}
void
nxt_sockaddr_cache_free(nxt_event_engine_t *engine, nxt_conn_t *c)
{
nxt_sockaddr_t *sa;
sa = c->remote;
nxt_event_engine_mem_free(engine, sa->cache_hint, sa, 0);
}
nxt_sockaddr_t *
nxt_sockaddr_alloc(nxt_mp_t *mp, socklen_t socklen, size_t address_length)
{
size_t size;
nxt_sockaddr_t *sa;
size = offsetof(nxt_sockaddr_t, u) + socklen + address_length;
/*
* The current struct sockaddr's define 32-bit fields at maximum
* and may define 64-bit AF_INET6 fields in the future. Alignment
* of memory allocated by nxt_mp_zalloc() is enough for these fields.
* If 128-bit alignment will be required then nxt_mem_malloc() and
* nxt_memzero() should be used instead.
*/
sa = nxt_mp_zalloc(mp, size);
if (nxt_fast_path(sa != NULL)) {
sa->socklen = socklen;
sa->length = address_length;
}
return sa;
}
nxt_sockaddr_t *
nxt_sockaddr_create(nxt_mp_t *mp, struct sockaddr *sockaddr, socklen_t length,
size_t address_length)
{
size_t size, copy;
nxt_sockaddr_t *sa;
size = length;
copy = length;
#if (NXT_HAVE_UNIX_DOMAIN)
/*
* Unspecified Unix domain sockaddr_un form and length are very
* platform depended (see comment in nxt_socket.h). Here they are
* normalized to the sockaddr_un with single zero byte sun_path[].
*/
if (size <= offsetof(struct sockaddr_un, sun_path)) {
/*
* Small socket length means a short unspecified Unix domain
* socket address:
*
* getsockname() and getpeername() on OpenBSD prior to 5.3
* return zero length and does not update a passed sockaddr
* buffer at all.
*
* Linux returns length equal to 2, i.e. sockaddr_un without
* sun_path[], unix(7):
*
* unnamed: A stream socket that has not been bound
* to a pathname using bind(2) has no name. Likewise,
* the two sockets created by socketpair(2) are unnamed.
* When the address of an unnamed socket is returned by
* getsockname(2), getpeername(2), and accept(2), its
* length is sizeof(sa_family_t), and sun_path should
* not be inspected.
*/
size = offsetof(struct sockaddr_un, sun_path) + 1;
#if !(NXT_LINUX)
} else if (sockaddr->sa_family == AF_UNIX && sockaddr->sa_data[0] == '\0') {
/*
* Omit nonsignificant zeros of the unspecified Unix domain socket
* address. This test is disabled for Linux since Linux abstract
* socket address also starts with zero. However Linux unspecified
* Unix domain socket address is short and is handled above.
*/
size = offsetof(struct sockaddr_un, sun_path) + 1;
copy = size;
#endif
}
#endif /* NXT_HAVE_UNIX_DOMAIN */
sa = nxt_sockaddr_alloc(mp, size, address_length);
if (nxt_fast_path(sa != NULL)) {
nxt_memcpy(&sa->u.sockaddr, sockaddr, copy);
#if (NXT_HAVE_UNIX_DOMAIN && NXT_OPENBSD)
if (length == 0) {
sa->u.sockaddr.sa_family = AF_UNIX;
}
#endif
}
return sa;
}
nxt_sockaddr_t *
nxt_sockaddr_copy(nxt_mp_t *mp, nxt_sockaddr_t *src)
{
size_t length;
nxt_sockaddr_t *dst;
length = offsetof(nxt_sockaddr_t, u) + src->socklen;
dst = nxt_mp_alloc(mp, length);
if (nxt_fast_path(dst != NULL)) {
nxt_memcpy(dst, src, length);
}
return dst;
}
nxt_sockaddr_t *
nxt_getsockname(nxt_task_t *task, nxt_mp_t *mp, nxt_socket_t s)
{
int ret;
size_t length;
socklen_t socklen;
nxt_sockaddr_buf_t sockaddr;
socklen = NXT_SOCKADDR_LEN;
ret = getsockname(s, &sockaddr.buf, &socklen);
if (nxt_fast_path(ret == 0)) {
switch (sockaddr.buf.sa_family) {
#if (NXT_INET6)
case AF_INET6:
length = NXT_INET6_ADDR_STR_LEN;
break;
#endif
#if (NXT_HAVE_UNIX_DOMAIN)
case AF_UNIX:
length = nxt_length("unix:") + socklen;
break;
#endif
case AF_INET:
length = NXT_INET_ADDR_STR_LEN;
break;
default:
length = 0;
break;
}
return nxt_sockaddr_create(mp, &sockaddr.buf, socklen, length);
}
nxt_log(task, NXT_LOG_ERR, "getsockname(%d) failed %E", s, nxt_errno);
return NULL;
}
void
nxt_sockaddr_text(nxt_sockaddr_t *sa)
{
size_t offset;
u_char *p, *start, *end, *octet;
uint32_t port;
offset = offsetof(nxt_sockaddr_t, u) + sa->socklen;
sa->start = offset;
sa->port_start = offset;
start = nxt_pointer_to(sa, offset);
end = start + sa->length;
switch (sa->u.sockaddr.sa_family) {
case AF_INET:
sa->address_start = offset;
octet = (u_char *) &sa->u.sockaddr_in.sin_addr;
p = nxt_sprintf(start, end, "%ud.%ud.%ud.%ud",
octet[0], octet[1], octet[2], octet[3]);
sa->address_length = p - start;
sa->port_start += sa->address_length + 1;
port = sa->u.sockaddr_in.sin_port;
break;
#if (NXT_INET6)
case AF_INET6:
sa->address_start = offset + 1;
p = start;
*p++ = '[';
p = nxt_inet6_ntop(sa->u.sockaddr_in6.sin6_addr.s6_addr, p, end);
sa->address_length = p - (start + 1);
sa->port_start += sa->address_length + 3;
*p++ = ']';
port = sa->u.sockaddr_in6.sin6_port;
break;
#endif
#if (NXT_HAVE_UNIX_DOMAIN)
case AF_UNIX:
sa->address_start = offset;
p = (u_char *) sa->u.sockaddr_un.sun_path;
#if (NXT_LINUX)
if (p[0] == '\0') {
size_t length;
/* Linux abstract socket address has no trailing zero. */
length = sa->socklen - offsetof(struct sockaddr_un, sun_path);
p = nxt_sprintf(start, end, "unix:@%*s", length - 1, p + 1);
} else {
p = nxt_sprintf(start, end, "unix:%s", p);
}
#else /* !(NXT_LINUX) */
p = nxt_sprintf(start, end, "unix:%s", p);
#endif
sa->address_length = p - start;
sa->port_start += sa->address_length;
sa->length = p - start;
return;
#endif /* NXT_HAVE_UNIX_DOMAIN */
default:
return;
}
p = nxt_sprintf(p, end, ":%d", ntohs(port));
sa->length = p - start;
}
uint32_t
nxt_sockaddr_port_number(nxt_sockaddr_t *sa)
{
uint32_t port;
switch (sa->u.sockaddr.sa_family) {
#if (NXT_INET6)
case AF_INET6:
port = sa->u.sockaddr_in6.sin6_port;
break;
#endif
#if (NXT_HAVE_UNIX_DOMAIN)
case AF_UNIX:
return 0;
#endif
default:
port = sa->u.sockaddr_in.sin_port;
break;
}
return ntohs((uint16_t) port);
}
nxt_bool_t
nxt_sockaddr_cmp(nxt_sockaddr_t *sa1, nxt_sockaddr_t *sa2)
{
if (sa1->socklen != sa2->socklen) {
return 0;
}
if (sa1->type != sa2->type) {
return 0;
}
if (sa1->u.sockaddr.sa_family != sa2->u.sockaddr.sa_family) {
return 0;
}
/*
* sockaddr struct's cannot be compared in whole since kernel
* may fill some fields in inherited sockaddr struct's.
*/
switch (sa1->u.sockaddr.sa_family) {
#if (NXT_INET6)
case AF_INET6:
if (sa1->u.sockaddr_in6.sin6_port != sa2->u.sockaddr_in6.sin6_port) {
return 0;
}
if (nxt_memcmp(&sa1->u.sockaddr_in6.sin6_addr,
&sa2->u.sockaddr_in6.sin6_addr, 16)
!= 0)
{
return 0;
}
return 1;
#endif
#if (NXT_HAVE_UNIX_DOMAIN)
case AF_UNIX:
{
size_t length;
length = sa1->socklen - offsetof(struct sockaddr_un, sun_path);
if (nxt_memcmp(&sa1->u.sockaddr_un.sun_path,
&sa2->u.sockaddr_un.sun_path, length)
!= 0)
{
return 0;
}
return 1;
}
#endif
default: /* AF_INET */
if (sa1->u.sockaddr_in.sin_port != sa2->u.sockaddr_in.sin_port) {
return 0;
}
if (sa1->u.sockaddr_in.sin_addr.s_addr
!= sa2->u.sockaddr_in.sin_addr.s_addr)
{
return 0;
}
return 1;
}
}
#if (NXT_INET6)
static u_char *
nxt_inet6_ntop(u_char *addr, u_char *buf, u_char *end)
{
u_char *p;
size_t zero_groups, last_zero_groups, ipv6_bytes;
nxt_uint_t i, zero_start, last_zero_start;
const size_t max_inet6_length =
nxt_length("ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff");
if (buf + max_inet6_length > end) {
return buf;
}
zero_start = 16;
zero_groups = 0;
last_zero_start = 16;
last_zero_groups = 0;
for (i = 0; i < 16; i += 2) {
if (addr[i] == 0 && addr[i + 1] == 0) {
if (last_zero_groups == 0) {
last_zero_start = i;
}
last_zero_groups++;
} else {
if (zero_groups < last_zero_groups) {
zero_groups = last_zero_groups;
zero_start = last_zero_start;
}
last_zero_groups = 0;
}
}
if (zero_groups < last_zero_groups) {
zero_groups = last_zero_groups;
zero_start = last_zero_start;
}
ipv6_bytes = 16;
p = buf;
if (zero_start == 0) {
/* IPv4-mapped address */
if ((zero_groups == 5 && addr[10] == 0xFF && addr[11] == 0xFF)
/* IPv4-compatible address */
|| (zero_groups == 6)
/* not IPv6 loopback address */
|| (zero_groups == 7 && addr[14] != 0 && addr[15] != 1))
{
ipv6_bytes = 12;
}
*p++ = ':';
}
for (i = 0; i < ipv6_bytes; i += 2) {
if (i == zero_start) {
/* Output maximum number of consecutive zero groups as "::". */
i += (zero_groups - 1) * 2;
*p++ = ':';
continue;
}
p = nxt_sprintf(p, end, "%uxd", (addr[i] << 8) + addr[i + 1]);
if (i < 14) {
*p++ = ':';
}
}
if (ipv6_bytes == 12) {
p = nxt_sprintf(p, end, "%ud.%ud.%ud.%ud",
addr[12], addr[13], addr[14], addr[15]);
}
return p;
}
#endif
nxt_sockaddr_t *
nxt_sockaddr_parse(nxt_mp_t *mp, nxt_str_t *addr)
{
nxt_sockaddr_t *sa;
sa = nxt_sockaddr_parse_optport(mp, addr);
if (sa != NULL
&& sa->u.sockaddr.sa_family != AF_UNIX
&& nxt_sockaddr_port_number(sa) == 0)
{
nxt_thread_log_error(NXT_LOG_ERR,
"The address \"%V\" must specify a port.", addr);
return NULL;
}
return sa;
}
nxt_sockaddr_t *
nxt_sockaddr_parse_optport(nxt_mp_t *mp, nxt_str_t *addr)
{
nxt_sockaddr_t *sa;
if (addr->length == 0) {
nxt_thread_log_error(NXT_LOG_ERR, "socket address cannot be empty");
return NULL;
}
if (addr->length > 6 && nxt_memcmp(addr->start, "unix:", 5) == 0) {
sa = nxt_sockaddr_unix_parse(mp, addr);
} else if (addr->start[0] == '[' || nxt_inet6_probe(addr)) {
sa = nxt_sockaddr_inet6_parse(mp, addr);
} else {
sa = nxt_sockaddr_inet_parse(mp, addr);
}
if (nxt_fast_path(sa != NULL)) {
nxt_sockaddr_text(sa);
}
return sa;
}
static nxt_sockaddr_t *
nxt_sockaddr_unix_parse(nxt_mp_t *mp, nxt_str_t *addr)
{
#if (NXT_HAVE_UNIX_DOMAIN)
size_t length, socklen;
u_char *path;
nxt_sockaddr_t *sa;
/*
* Actual sockaddr_un length can be lesser or even larger than defined
* struct sockaddr_un length (see comment in nxt_socket.h). So
* limit maximum Unix domain socket address length by defined sun_path[]
* length because some OSes accept addresses twice larger than defined
* struct sockaddr_un. Also reserve space for a trailing zero to avoid
* ambiguity, since many OSes accept Unix domain socket addresses
* without a trailing zero.
*/
const size_t max_len = sizeof(struct sockaddr_un)
- offsetof(struct sockaddr_un, sun_path) - 1;
/* Cutting "unix:". */
length = addr->length - 5;
path = addr->start + 5;
if (length > max_len) {
nxt_thread_log_error(NXT_LOG_ERR,
"unix domain socket \"%V\" name is too long",
addr);
return NULL;
}
socklen = offsetof(struct sockaddr_un, sun_path) + length + 1;
#if (NXT_LINUX)
/*
* Linux unix(7):
*
* abstract: an abstract socket address is distinguished by the fact
* that sun_path[0] is a null byte ('\0'). The socket's address in
* this namespace is given by the additional bytes in sun_path that
* are covered by the specified length of the address structure.
* (Null bytes in the name have no special significance.)
*/
if (path[0] == '@') {
path[0] = '\0';
socklen--;
}
#endif
sa = nxt_sockaddr_alloc(mp, socklen, addr->length);
if (nxt_fast_path(sa != NULL)) {
sa->u.sockaddr_un.sun_family = AF_UNIX;
nxt_memcpy(sa->u.sockaddr_un.sun_path, path, length);
}
return sa;
#else /* !(NXT_HAVE_UNIX_DOMAIN) */
nxt_thread_log_error(NXT_LOG_ERR,
"unix domain socket \"%V\" is not supported", addr);
return NULL;
#endif
}
static nxt_sockaddr_t *
nxt_sockaddr_inet6_parse(nxt_mp_t *mp, nxt_str_t *addr)
{
#if (NXT_INET6)
u_char *p, *start, *end;
size_t length;
nxt_int_t ret, port;
nxt_sockaddr_t *sa;
if (addr->start[0] == '[') {
length = addr->length - 1;
start = addr->start + 1;
end = nxt_memchr(start, ']', length);
if (nxt_slow_path(end == NULL)) {
return NULL;
}
p = end + 1;
} else {
length = addr->length;
start = addr->start;
end = addr->start + addr->length;
p = NULL;
}
port = 0;
if (p != NULL) {
length = (start + length) - p;
if (length < 2 || *p != ':') {
nxt_thread_log_error(NXT_LOG_ERR, "invalid IPv6 address in \"%V\"",
addr);
return NULL;
}
port = nxt_int_parse(p + 1, length - 1);
if (port < 1 || port > 65535) {
nxt_thread_log_error(NXT_LOG_ERR, "invalid port in \"%V\"", addr);
return NULL;
}
}
sa = nxt_sockaddr_alloc(mp, sizeof(struct sockaddr_in6),
NXT_INET6_ADDR_STR_LEN);
if (nxt_slow_path(sa == NULL)) {
return NULL;
}
ret = nxt_inet6_addr(&sa->u.sockaddr_in6.sin6_addr, start, end - start);
if (nxt_slow_path(ret != NXT_OK)) {
nxt_thread_log_error(NXT_LOG_ERR, "invalid IPv6 address in \"%V\"",
addr);
return NULL;
}
sa->u.sockaddr_in6.sin6_family = AF_INET6;
sa->u.sockaddr_in6.sin6_port = htons((in_port_t) port);
return sa;
#else /* !(NXT_INET6) */
nxt_thread_log_error(NXT_LOG_ERR, "IPv6 socket \"%V\" is not supported",
addr);
return NULL;
#endif
}
static nxt_sockaddr_t *
nxt_sockaddr_inet_parse(nxt_mp_t *mp, nxt_str_t *addr)
{
u_char *p;
size_t length;
nxt_int_t port;
in_addr_t inaddr;
nxt_sockaddr_t *sa;
p = nxt_memchr(addr->start, ':', addr->length);
if (p == NULL) {
length = addr->length;
} else {
length = p - addr->start;
}
inaddr = INADDR_ANY;
if (length != 1 || addr->start[0] != '*') {
inaddr = nxt_inet_addr(addr->start, length);
if (nxt_slow_path(inaddr == INADDR_NONE)) {
nxt_thread_log_error(NXT_LOG_ERR, "invalid address \"%V\"", addr);
return NULL;
}
}
port = 0;
if (p != NULL) {
p++;
length = (addr->start + addr->length) - p;
port = nxt_int_parse(p, length);
if (port < 1 || port > 65535) {
nxt_thread_log_error(NXT_LOG_ERR, "invalid port in \"%V\"", addr);
return NULL;
}
}
sa = nxt_sockaddr_alloc(mp, sizeof(struct sockaddr_in),
NXT_INET_ADDR_STR_LEN);
if (nxt_slow_path(sa == NULL)) {
return NULL;
}
sa->u.sockaddr_in.sin_family = AF_INET;
sa->u.sockaddr_in.sin_addr.s_addr = inaddr;
sa->u.sockaddr_in.sin_port = htons((in_port_t) port);
return sa;
}
in_addr_t
nxt_inet_addr(u_char *buf, size_t length)
{
u_char c, *end;
in_addr_t addr;
nxt_uint_t digit, octet, dots;
if (nxt_slow_path(*(buf + length - 1) == '.')) {
return INADDR_NONE;
}
addr = 0;
octet = 0;
dots = 0;
end = buf + length;
while (buf < end) {
c = *buf++;
digit = c - '0';
/* values below '0' become large unsigned integers */
if (digit < 10) {
octet = octet * 10 + digit;
continue;
}
if (c == '.' && octet < 256) {
addr = (addr << 8) + octet;
octet = 0;
dots++;
continue;
}
return INADDR_NONE;
}
if (dots == 3 && octet < 256) {
addr = (addr << 8) + octet;
return htonl(addr);
}
return INADDR_NONE;
}
#if (NXT_INET6)
nxt_int_t
nxt_inet6_addr(struct in6_addr *in6_addr, u_char *buf, size_t length)
{
u_char c, *addr, *zero_start, *ipv4, *dst, *src, *end;
nxt_uint_t digit, group, nibbles, groups_left;
if (length == 0) {
return NXT_ERROR;
}
end = buf + length;
if (buf[0] == ':') {
buf++;
}
addr = in6_addr->s6_addr;
zero_start = NULL;
groups_left = 8;
nibbles = 0;
group = 0;
ipv4 = NULL;
while (buf < end) {
c = *buf++;
if (c == ':') {
if (nibbles != 0) {
ipv4 = buf;
*addr++ = (u_char) (group >> 8);
*addr++ = (u_char) (group & 0xFF);
groups_left--;
if (groups_left != 0) {
nibbles = 0;
group = 0;
continue;
}
} else {
if (zero_start == NULL) {
ipv4 = buf;
zero_start = addr;
continue;
}
}
return NXT_ERROR;
}
if (c == '.' && nibbles != 0) {
if (groups_left < 2 || ipv4 == NULL) {
return NXT_ERROR;
}
group = nxt_inet_addr(ipv4, end - ipv4);
if (group == INADDR_NONE) {
return NXT_ERROR;
}
group = ntohl(group);
*addr++ = (u_char) ((group >> 24) & 0xFF);
*addr++ = (u_char) ((group >> 16) & 0xFF);
groups_left--;
/* the low 16-bit are copied below */
break;
}
nibbles++;
if (nibbles > 4) {
return NXT_ERROR;
}
group <<= 4;
digit = c - '0';
/* values below '0' become large unsigned integers */
if (digit < 10) {
group += digit;
continue;
}
c |= 0x20;
digit = c - 'a';
/* values below 'a' become large unsigned integers */
if (digit < 6) {
group += 10 + digit;
continue;
}
return NXT_ERROR;
}
if (nibbles == 0 && zero_start == NULL) {
return NXT_ERROR;
}
*addr++ = (u_char) (group >> 8);
*addr++ = (u_char) (group & 0xFF);
groups_left--;
if (groups_left != 0) {
if (zero_start != NULL) {
/* moving part before consecutive zero groups to the end */
groups_left *= 2;
src = addr - 1;
dst = src + groups_left;
while (src >= zero_start) {
*dst-- = *src--;
}
nxt_memzero(zero_start, groups_left);
return NXT_OK;
}
} else {
if (zero_start == NULL) {
return NXT_OK;
}
}
return NXT_ERROR;
}
#endif
nxt_bool_t
nxt_inet6_probe(nxt_str_t *str)
{
u_char *colon, *end;
colon = nxt_memchr(str->start, ':', str->length);
if (colon != NULL) {
end = str->start + str->length;
colon = nxt_memchr(colon + 1, ':', end - (colon + 1));
}
return (colon != NULL);
}