tcp_address.cpp 17.2 KB
Newer Older
1
/*
2
    Copyright (c) 2007-2013 Contributors as noted in the AUTHORS file
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

    This file is part of 0MQ.

    0MQ is free software; you can redistribute it and/or modify it under
    the terms of the GNU Lesser General Public License as published by
    the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.

    0MQ is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Lesser General Public License for more details.

    You should have received a copy of the GNU Lesser General Public License
    along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

#include <string>
21
#include <sstream>
22 23 24

#include "tcp_address.hpp"
#include "platform.hpp"
Martin Sustrik's avatar
Martin Sustrik committed
25
#include "stdint.hpp"
26
#include "err.hpp"
27
#include "ip.hpp"
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49

#ifdef ZMQ_HAVE_WINDOWS
#include "windows.hpp"
#else
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/tcp.h>
#include <netdb.h>
#endif

//  Some platforms (notably Darwin/OSX and NetBSD) do not define all AI_
//  flags for getaddrinfo(). This can be worked around safely by defining
//  these to 0.
#ifndef AI_ADDRCONFIG
#define AI_ADDRCONFIG 0
#endif

#if defined ZMQ_HAVE_SOLARIS

#include <sys/sockio.h>
#include <net/if.h>
#include <unistd.h>
50
#include <stdlib.h>
51 52

//  On Solaris platform, network interface name can be queried by ioctl.
Pieter Hintjens's avatar
Pieter Hintjens committed
53
int zmq::tcp_address_t::resolve_nic_name (const char *nic_, bool ipv6_)
54 55
{
    //  TODO: Unused parameter, IPv6 support not implemented for Solaris.
Pieter Hintjens's avatar
Pieter Hintjens committed
56
    (void) ipv6_;
57 58

    //  Create a socket.
59
    int fd = open_socket (AF_INET, SOCK_DGRAM, 0);
60
    errno_assert (fd != -1);
61 62 63 64 65 66

    //  Retrieve number of interfaces.
    lifnum ifn;
    ifn.lifn_family = AF_INET;
    ifn.lifn_flags = 0;
    int rc = ioctl (fd, SIOCGLIFNUM, (char*) &ifn);
67
    errno_assert (rc != -1);
68 69 70 71 72

    //  Allocate memory to get interface names.
    size_t ifr_size = sizeof (struct lifreq) * ifn.lifn_count;
    char *ifr = (char*) malloc (ifr_size);
    alloc_assert (ifr);
73

74 75 76 77 78 79 80
    //  Retrieve interface names.
    lifconf ifc;
    ifc.lifc_family = AF_INET;
    ifc.lifc_flags = 0;
    ifc.lifc_len = ifr_size;
    ifc.lifc_buf = ifr;
    rc = ioctl (fd, SIOCGLIFCONF, (char*) &ifc);
81
    errno_assert (rc != -1);
82 83 84 85 86 87

    //  Find the interface with the specified name and AF_INET family.
    bool found = false;
    lifreq *ifrp = ifc.lifc_req;
    for (int n = 0; n < (int) (ifc.lifc_len / sizeof (lifreq));
          n ++, ifrp ++) {
88
        if (!strcmp (nic_, ifrp->lifr_name)) {
89
            rc = ioctl (fd, SIOCGLIFADDR, (char*) ifrp);
90
            errno_assert (rc != -1);
91
            if (ifrp->lifr_addr.ss_family == AF_INET) {
92
                address.ipv4 = *(sockaddr_in*) &ifrp->lifr_addr;
93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
                found = true;
                break;
            }
        }
    }

    //  Clean-up.
    free (ifr);
    close (fd);

    if (!found) {
        errno = ENODEV;
        return -1;
    }
    return 0;
}

110
#elif defined ZMQ_HAVE_AIX || defined ZMQ_HAVE_HPUX || defined ZMQ_HAVE_ANDROID
111 112 113 114 115 116

#include <sys/types.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <net/if.h>

Pieter Hintjens's avatar
Pieter Hintjens committed
117
int zmq::tcp_address_t::resolve_nic_name (const char *nic_, bool ipv6_)
118 119
{
    //  TODO: Unused parameter, IPv6 support not implemented for AIX or HP/UX.
Pieter Hintjens's avatar
Pieter Hintjens committed
120
    (void) ipv6_;
121 122

    //  Create a socket.
123
    int sd = open_socket (AF_INET, SOCK_DGRAM, 0);
124
    errno_assert (sd != -1);
125

126
    struct ifreq ifr;
127 128

    //  Copy interface name for ioctl get.
129
    strncpy (ifr.ifr_name, nic_, sizeof (ifr.ifr_name));
130 131 132 133 134 135 136 137 138 139 140

    //  Fetch interface address.
    int rc = ioctl (sd, SIOCGIFADDR, (caddr_t) &ifr, sizeof (struct ifreq));

    //  Clean up.
    close (sd);

    if (rc == -1) {
        errno = ENODEV;
        return -1;
    }
141 142
    memcpy (&address.ipv4.sin_addr, &((sockaddr_in*) &ifr.ifr_addr)->sin_addr,
        sizeof (in_addr));
143

144
    return 0;
145 146 147 148 149 150 151 152 153 154 155
}

#elif ((defined ZMQ_HAVE_LINUX || defined ZMQ_HAVE_FREEBSD ||\
    defined ZMQ_HAVE_OSX || defined ZMQ_HAVE_OPENBSD ||\
    defined ZMQ_HAVE_QNXNTO || defined ZMQ_HAVE_NETBSD)\
    && defined ZMQ_HAVE_IFADDRS)

#include <ifaddrs.h>

//  On these platforms, network interface name can be queried
//  using getifaddrs function.
Pieter Hintjens's avatar
Pieter Hintjens committed
156
int zmq::tcp_address_t::resolve_nic_name (const char *nic_, bool ipv6_)
157 158
{
    //  Get the addresses.
Pieter Hintjens's avatar
Pieter Hintjens committed
159
    ifaddrs *ifa = NULL;
160
    int rc = getifaddrs (&ifa);
161
    errno_assert (rc == 0);
162 163 164 165 166 167 168 169 170 171
    zmq_assert (ifa != NULL);

    //  Find the corresponding network interface.
    bool found = false;
    for (ifaddrs *ifp = ifa; ifp != NULL ;ifp = ifp->ifa_next)
    {
        if (ifp->ifa_addr == NULL)
            continue;

        int family = ifp->ifa_addr->sa_family;
Pieter Hintjens's avatar
Pieter Hintjens committed
172 173
        if ((family == AF_INET || (ipv6_ && family == AF_INET6))
        && !strcmp (nic_, ifp->ifa_name)) {
174
            memcpy (&address, ifp->ifa_addr,
175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
                    (family == AF_INET) ? sizeof (struct sockaddr_in)
                                        : sizeof (struct sockaddr_in6));
            found = true;
            break;
        }
    }

    //  Clean-up;
    freeifaddrs (ifa);

    if (!found) {
        errno = ENODEV;
        return -1;
    }
    return 0;
}

#else

//  On other platforms we assume there are no sane interface names.
//  This is true especially of Windows.
Pieter Hintjens's avatar
Pieter Hintjens committed
196
int zmq::tcp_address_t::resolve_nic_name (const char *nic_, bool ipv6_)
197 198
{
    //  All unused parameters.
199
    (void) nic_;
Pieter Hintjens's avatar
Pieter Hintjens committed
200
    (void) ipv6_;
201 202 203 204 205 206 207

    errno = ENODEV;
    return -1;
}

#endif

Pieter Hintjens's avatar
Pieter Hintjens committed
208
int zmq::tcp_address_t::resolve_interface (const char *interface_, bool ipv6_)
209 210 211
{
    //  Initialize temporary output pointers with storage address.
    sockaddr_storage ss;
212 213
    sockaddr *out_addr = (sockaddr*) &ss;
    size_t out_addrlen;
214 215 216

    //  Initialise IP-format family/port and populate temporary output pointers
    //  with the address.
Pieter Hintjens's avatar
Pieter Hintjens committed
217
    if (ipv6_) {
218 219 220 221
        sockaddr_in6 ip6_addr;
        memset (&ip6_addr, 0, sizeof (ip6_addr));
        ip6_addr.sin6_family = AF_INET6;
        memcpy (&ip6_addr.sin6_addr, &in6addr_any, sizeof (in6addr_any));
222
        out_addrlen = sizeof ip6_addr;
223 224
        memcpy (out_addr, &ip6_addr, out_addrlen);
    }
Pieter Hintjens's avatar
Pieter Hintjens committed
225 226 227 228 229 230 231 232 233
    else {
        sockaddr_in ip4_addr;
        memset (&ip4_addr, 0, sizeof (ip4_addr));
        ip4_addr.sin_family = AF_INET;
        ip4_addr.sin_addr.s_addr = htonl (INADDR_ANY);
        out_addrlen = sizeof ip4_addr;
        memcpy (out_addr, &ip4_addr, out_addrlen);
    }
    //  "*" resolves to INADDR_ANY or in6addr_any.
234
    if (strcmp (interface_, "*") == 0) {
235
        zmq_assert (out_addrlen <= sizeof address);
236 237 238 239 240
        memcpy (&address, out_addr, out_addrlen);
        return 0;
    }

    //  Try to resolve the string as a NIC name.
Pieter Hintjens's avatar
Pieter Hintjens committed
241
    int rc = resolve_nic_name (interface_, ipv6_);
242 243
    if (rc != 0 && errno != ENODEV)
        return rc;
244
    if (rc == 0)
245 246 247 248 249 250 251 252 253 254 255 256 257 258
        return 0;

    //  There's no such interface name. Assume literal address.
#if defined ZMQ_HAVE_OPENVMS && defined __ia64
    __addrinfo64 *res = NULL;
    __addrinfo64 req;
#else
    addrinfo *res = NULL;
    addrinfo req;
#endif
    memset (&req, 0, sizeof (req));

    //  Choose IPv4 or IPv6 protocol family. Note that IPv6 allows for
    //  IPv4-in-IPv6 addresses.
Pieter Hintjens's avatar
Pieter Hintjens committed
259
    req.ai_family = ipv6_? AF_INET6: AF_INET;
260 261 262 263 264 265 266 267

    //  Arbitrary, not used in the output, but avoids duplicate results.
    req.ai_socktype = SOCK_STREAM;

    //  Restrict hostname/service to literals to avoid any DNS lookups or
    //  service-name irregularity due to indeterminate socktype.
    req.ai_flags = AI_PASSIVE | AI_NUMERICHOST;

268 269 270 271 272 273
#if defined AI_V4MAPPED && !defined ZMQ_HAVE_FREEBSD
    //  In this API we only require IPv4-mapped addresses when
    //  no native IPv6 interfaces are available (~AI_ALL).
    //  This saves an additional DNS roundtrip for IPv4 addresses.
    //  Note: While the AI_V4MAPPED flag is defined on FreeBSD system,
    //  it is not supported here. See libzmq issue #331.
274 275 276 277 278 279 280 281 282 283 284 285 286
    if (req.ai_family == AF_INET6)
        req.ai_flags |= AI_V4MAPPED;
#endif

    //  Resolve the literal address. Some of the error info is lost in case
    //  of error, however, there's no way to report EAI errors via errno.
    rc = getaddrinfo (interface_, NULL, &req, &res);
    if (rc) {
        errno = ENODEV;
        return -1;
    }

    //  Use the first result.
287
    zmq_assert (res != NULL);
288 289 290 291
    zmq_assert ((size_t) (res->ai_addrlen) <= sizeof (address));
    memcpy (&address, res->ai_addr, res->ai_addrlen);

    //  Cleanup getaddrinfo after copying the possibly referenced result.
292
    freeaddrinfo (res);
293 294 295 296

    return 0;
}

Pieter Hintjens's avatar
Pieter Hintjens committed
297
int zmq::tcp_address_t::resolve_hostname (const char *hostname_, bool ipv6_)
298 299
{
    //  Set up the query.
300 301 302
#if defined ZMQ_HAVE_OPENVMS && defined __ia64 && __INITIAL_POINTER_SIZE == 64
    __addrinfo64 req;
#else
303
    addrinfo req;
304
#endif
305 306 307 308
    memset (&req, 0, sizeof (req));

    //  Choose IPv4 or IPv6 protocol family. Note that IPv6 allows for
    //  IPv4-in-IPv6 addresses.
Pieter Hintjens's avatar
Pieter Hintjens committed
309
    req.ai_family = ipv6_? AF_INET6: AF_INET;
310 311 312 313

    //  Need to choose one to avoid duplicate results from getaddrinfo() - this
    //  doesn't really matter, since it's not included in the addr-output.
    req.ai_socktype = SOCK_STREAM;
314 315 316 317 318 319 320

#if defined AI_V4MAPPED && !defined ZMQ_HAVE_FREEBSD
    //  In this API we only require IPv4-mapped addresses when
    //  no native IPv6 interfaces are available.
    //  This saves an additional DNS roundtrip for IPv4 addresses.
    //  Note: While the AI_V4MAPPED flag is defined on FreeBSD system,
    //  it is not supported here. See libzmq issue #331.
321 322 323 324 325 326
    if (req.ai_family == AF_INET6)
        req.ai_flags |= AI_V4MAPPED;
#endif

    //  Resolve host name. Some of the error info is lost in case of error,
    //  however, there's no way to report EAI errors via errno.
327 328 329
#if defined ZMQ_HAVE_OPENVMS && defined __ia64 && __INITIAL_POINTER_SIZE == 64
    __addrinfo64 *res;
#else
330
    addrinfo *res;
331
#endif
332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
    int rc = getaddrinfo (hostname_, NULL, &req, &res);
    if (rc) {
        switch (rc) {
        case EAI_MEMORY:
            errno = ENOMEM;
            break;
        default:
            errno = EINVAL;
            break;
        }
        return -1;
    }

    //  Copy first result to output addr with hostname and service.
    zmq_assert ((size_t) (res->ai_addrlen) <= sizeof (address));
    memcpy (&address, res->ai_addr, res->ai_addrlen);
348

349
    freeaddrinfo (res);
350

351 352 353 354 355 356 357 358
    return 0;
}

zmq::tcp_address_t::tcp_address_t ()
{
    memset (&address, 0, sizeof (address));
}

359 360 361 362 363
zmq::tcp_address_t::tcp_address_t (const sockaddr *sa, socklen_t sa_len)
{
    zmq_assert(sa && sa_len > 0);

    memset (&address, 0, sizeof (address));
364
    if (sa->sa_family == AF_INET && sa_len >= (socklen_t) sizeof (address.ipv4))
365
        memcpy(&address.ipv4, sa, sizeof (address.ipv4));
366
    else 
367
    if (sa->sa_family == AF_INET6 && sa_len >= (socklen_t) sizeof (address.ipv6))
368 369 370
        memcpy(&address.ipv6, sa, sizeof (address.ipv6));
}

371 372 373 374
zmq::tcp_address_t::~tcp_address_t ()
{
}

Pieter Hintjens's avatar
Pieter Hintjens committed
375
int zmq::tcp_address_t::resolve (const char *name_, bool local_, bool ipv6_)
376 377 378 379 380 381 382 383 384 385 386
{
    //  Find the ':' at end that separates address from the port number.
    const char *delimiter = strrchr (name_, ':');
    if (!delimiter) {
        errno = EINVAL;
        return -1;
    }
    //  Separate the address/port.
    std::string addr_str (name_, delimiter - name_);
    std::string port_str (delimiter + 1);

387
    //  Remove square brackets around the address, if any.
388
    if (addr_str.size () >= 2 && addr_str [0] == '[' &&
389 390 391
          addr_str [addr_str.size () - 1] == ']')
        addr_str = addr_str.substr (1, addr_str.size () - 2);

392
    //  Allow 0 specifically, to detect invalid port error in atoi if not
Pieter Hintjens's avatar
Pieter Hintjens committed
393
    uint16_t port;
394
    if (port_str == "*" || port_str == "0")
395
        //  Resolve wildcard to 0 to allow autoselection of port
396
        port = 0;
397
    else {
398
        //  Parse the port number (0 is not a valid port).
399
        port = (uint16_t) atoi (port_str.c_str ());
400 401 402 403
        if (port == 0) {
            errno = EINVAL;
            return -1;
        }
404 405 406 407 408
    }

    //  Resolve the IP address.
    int rc;
    if (local_)
Pieter Hintjens's avatar
Pieter Hintjens committed
409
        rc = resolve_interface (addr_str.c_str (), ipv6_);
410
    else
Pieter Hintjens's avatar
Pieter Hintjens committed
411
        rc = resolve_hostname (addr_str.c_str (), ipv6_);
412 413 414 415 416 417 418 419 420 421 422 423
    if (rc != 0)
        return -1;

    //  Set the port into the address structure.
    if (address.generic.sa_family == AF_INET6)
        address.ipv6.sin6_port = htons (port);
    else
        address.ipv4.sin_port = htons (port);

    return 0;
}

424 425
int zmq::tcp_address_t::to_string (std::string &addr_)
{
Pieter Hintjens's avatar
Pieter Hintjens committed
426 427
    if (address.generic.sa_family != AF_INET
    &&  address.generic.sa_family != AF_INET6) {
428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452
        addr_.clear ();
        return -1;
    }

    // not using service resolv because of https://github.com/zeromq/libzmq/commit/1824574f9b5a8ce786853320e3ea09fe1f822bc4
    char hbuf[NI_MAXHOST];
    int rc = getnameinfo (addr (), addrlen (), hbuf, sizeof (hbuf), NULL, 0, NI_NUMERICHOST);
    if (rc != 0) {
        addr_.clear ();
        return rc;
    }

    if (address.generic.sa_family == AF_INET6) {
        std::stringstream s;
        s << "tcp://[" << hbuf << "]:" << ntohs (address.ipv6.sin6_port);
        addr_ = s.str ();
    }
    else {
        std::stringstream s;
        s << "tcp://" << hbuf << ":" << ntohs (address.ipv4.sin_port);
        addr_ = s.str ();
    };
    return 0;
}

453
const sockaddr *zmq::tcp_address_t::addr () const
454 455 456 457
{
    return &address.generic;
}

458
socklen_t zmq::tcp_address_t::addrlen () const
459 460 461 462 463 464 465
{
    if (address.generic.sa_family == AF_INET6)
        return (socklen_t) sizeof (address.ipv6);
    else
        return (socklen_t) sizeof (address.ipv4);
}

Martin Sustrik's avatar
Martin Sustrik committed
466
#if defined ZMQ_HAVE_WINDOWS
467
unsigned short zmq::tcp_address_t::family () const
Martin Sustrik's avatar
Martin Sustrik committed
468
#else
469
sa_family_t zmq::tcp_address_t::family () const
Martin Sustrik's avatar
Martin Sustrik committed
470
#endif
471 472 473 474
{
    return address.generic.sa_family;
}

475 476 477 478 479 480
zmq::tcp_address_mask_t::tcp_address_mask_t () :
    tcp_address_t ()
{
    address_mask = -1;
}

481
int zmq::tcp_address_mask_t::mask () const
482 483 484 485
{
    return address_mask;
}

Pieter Hintjens's avatar
Pieter Hintjens committed
486
int zmq::tcp_address_mask_t::resolve (const char *name_, bool ipv6_)
487 488 489
{
    // Find '/' at the end that separates address from the cidr mask number.
    // Allow empty mask clause and threat it like '/32' for ipv4 or '/128' for ipv6.
490
    std::string addr_str, mask_str;
491 492 493 494 495 496 497 498 499
    const char *delimiter = strrchr (name_, '/');
    if (delimiter != NULL) {
        addr_str.assign (name_, delimiter - name_);
        mask_str.assign (delimiter + 1);
        if (mask_str.empty ()) {
            errno = EINVAL;
            return -1;
        }
    }
Pieter Hintjens's avatar
Pieter Hintjens committed
500
    else
501 502 503
        addr_str.assign (name_);

    // Parse address part using standard routines.
Pieter Hintjens's avatar
Pieter Hintjens committed
504
    int rc = tcp_address_t::resolve_hostname (addr_str.c_str (), ipv6_);
505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534
    if (rc != 0)
        return rc;

    // Parse the cidr mask number.
    if (mask_str.empty ()) {
        if (address.generic.sa_family == AF_INET6)
            address_mask = 128;
        else
            address_mask = 32;
    }
    else
    if (mask_str == "0") {
        address_mask = 0;
    }
    else {
        int mask = atoi (mask_str.c_str ());
        if (
            (mask < 1) ||
            (address.generic.sa_family == AF_INET6 && mask > 128) ||
            (address.generic.sa_family != AF_INET6 && mask > 32)
        ) {
            errno = EINVAL;
            return -1;
        }
        address_mask = mask;
    }

    return 0;
}

535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565
int zmq::tcp_address_mask_t::to_string (std::string &addr_)
{
    if (address.generic.sa_family != AF_INET && address.generic.sa_family != AF_INET6) {
        addr_.clear ();
        return -1;
    }
    if (address_mask == -1) {
        addr_.clear ();
        return -1;
    }

    char hbuf[NI_MAXHOST];
    int rc = getnameinfo (addr (), addrlen (), hbuf, sizeof (hbuf), NULL, 0, NI_NUMERICHOST);
    if (rc != 0) {
        addr_.clear ();
        return rc;
    }

    if (address.generic.sa_family == AF_INET6) {
        std::stringstream s;
        s << "[" << hbuf << "]/" << address_mask;
        addr_ = s.str ();
    }
    else {
        std::stringstream s;
        s << hbuf << "/" << address_mask;
        addr_ = s.str ();
    };
    return 0;
}

566
bool zmq::tcp_address_mask_t::match_address (const struct sockaddr *ss, const socklen_t ss_len) const
567
{
568
    zmq_assert (address_mask != -1 && ss != NULL && ss_len >= (socklen_t) sizeof (struct sockaddr));
569 570 571 572 573 574

    if (ss->sa_family != address.generic.sa_family)
        return false;

    if (address_mask > 0) {
        int mask;
575
        const uint8_t *our_bytes, *their_bytes;
576 577
        if (ss->sa_family == AF_INET6) {
            zmq_assert (ss_len == sizeof (struct sockaddr_in6));
578 579
            their_bytes = (const uint8_t *) &(((const struct sockaddr_in6 *) ss)->sin6_addr);
            our_bytes = (const uint8_t *) &address.ipv6.sin6_addr;
580 581 582 583
            mask = sizeof (struct in6_addr) * 8;
        }
        else {
            zmq_assert (ss_len == sizeof (struct sockaddr_in));
584 585
            their_bytes = (const uint8_t *) &(((const struct sockaddr_in *) ss)->sin_addr);
            our_bytes = (const uint8_t *) &address.ipv4.sin_addr;
586 587 588 589
            mask = sizeof (struct in_addr) * 8;
        }
        if (address_mask < mask) mask = address_mask;

590 591 592
        size_t full_bytes = mask / 8;
        if (memcmp(our_bytes, their_bytes, full_bytes))
            return false;
593

594
        uint8_t last_byte_bits = (0xffU << (8 - (mask % 8))) & 0xffU;
595 596 597 598 599 600 601 602
        if (last_byte_bits) {
            if ((their_bytes[full_bytes] & last_byte_bits) != (our_bytes[full_bytes] & last_byte_bits))
                return false;
        }
    }

    return true;
}