aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/transport/if_addrs.cpp
blob: a1cb6909f03429900b39adfe658c1e2544e7641b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
//
// Copyright 2010-2011 Ettus Research LLC
// Copyright 2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0-or-later
//

#include <uhd/transport/if_addrs.hpp>
#include <stdint.h>
#include <boost/asio/ip/address_v4.hpp>
#include <iostream>

/***********************************************************************
 * Interface address discovery through ifaddrs api
 **********************************************************************/
#ifdef HAVE_GETIFADDRS
#    include <ifaddrs.h>

static boost::asio::ip::address_v4 sockaddr_to_ip_addr(sockaddr* addr)
{
    return boost::asio::ip::address_v4(
        ntohl(reinterpret_cast<sockaddr_in*>(addr)->sin_addr.s_addr));
}

std::vector<uhd::transport::if_addrs_t> uhd::transport::get_if_addrs(void)
{
    std::vector<if_addrs_t> if_addrs;
    struct ifaddrs* ifap;
    if (getifaddrs(&ifap) == 0) {
        for (struct ifaddrs* iter = ifap; iter != nullptr; iter = iter->ifa_next) {
            // ensure that the entries are valid
            if (iter->ifa_addr == nullptr)
                continue;
            if (iter->ifa_addr->sa_family != AF_INET)
                continue;
            if (iter->ifa_netmask->sa_family != AF_INET)
                continue;
            if (iter->ifa_broadaddr->sa_family != AF_INET)
                continue;

            // append a new set of interface addresses
            if_addrs_t if_addr;
            if_addr.inet  = sockaddr_to_ip_addr(iter->ifa_addr).to_string();
            if_addr.mask  = sockaddr_to_ip_addr(iter->ifa_netmask).to_string();
            if_addr.bcast = sockaddr_to_ip_addr(iter->ifa_broadaddr).to_string();

            // correct the bcast address when its same as the gateway
            if (if_addr.inet == if_addr.bcast
                or sockaddr_to_ip_addr(iter->ifa_broadaddr)
                       == boost::asio::ip::address_v4(0)) {
                // manually calculate broadcast address
                // https://svn.boost.org/trac/boost/ticket/5198
                const uint32_t addr  = sockaddr_to_ip_addr(iter->ifa_addr).to_ulong();
                const uint32_t mask  = sockaddr_to_ip_addr(iter->ifa_netmask).to_ulong();
                const uint32_t bcast = (addr & mask) | ~mask;
                if_addr.bcast        = boost::asio::ip::address_v4(bcast).to_string();
            }

            if_addrs.push_back(if_addr);
        }
        freeifaddrs(ifap);
    }
    return if_addrs;
}

#endif /* HAVE_GETIFADDRS */

/***********************************************************************
 * Interface address discovery through windows api
 **********************************************************************/
#ifdef HAVE_SIO_GET_INTERFACE_LIST
#    include <winsock2.h>

std::vector<uhd::transport::if_addrs_t> uhd::transport::get_if_addrs(void)
{
    std::vector<if_addrs_t> if_addrs;
    SOCKET sd = WSASocket(AF_INET, SOCK_DGRAM, 0, 0, 0, 0);
    if (sd == SOCKET_ERROR) {
        std::cerr << "Failed to get a socket. Error " << WSAGetLastError() << std::endl;
        return if_addrs;
    }

    INTERFACE_INFO InterfaceList[20];
    unsigned long nBytesReturned;
    if (WSAIoctl(sd,
            SIO_GET_INTERFACE_LIST,
            0,
            0,
            &InterfaceList,
            sizeof(InterfaceList),
            &nBytesReturned,
            0,
            0)
        == SOCKET_ERROR) {
        std::cerr << "Failed calling WSAIoctl: error " << WSAGetLastError() << std::endl;
        return if_addrs;
    }

    int nNumInterfaces = nBytesReturned / sizeof(INTERFACE_INFO);
    for (int i = 0; i < nNumInterfaces; ++i) {
        uint32_t iiAddress = ntohl(
            reinterpret_cast<sockaddr_in&>(InterfaceList[i].iiAddress).sin_addr.s_addr);
        uint32_t iiNetmask = ntohl(
            reinterpret_cast<sockaddr_in&>(InterfaceList[i].iiNetmask).sin_addr.s_addr);
        uint32_t iiBroadcastAddress = (iiAddress & iiNetmask) | ~iiNetmask;

        if_addrs_t if_addr;
        if_addr.inet  = boost::asio::ip::address_v4(iiAddress).to_string();
        if_addr.mask  = boost::asio::ip::address_v4(iiNetmask).to_string();
        if_addr.bcast = boost::asio::ip::address_v4(iiBroadcastAddress).to_string();
        if_addrs.push_back(if_addr);
    }

    return if_addrs;
}

#endif /* HAVE_SIO_GET_INTERFACE_LIST */

/***********************************************************************
 * Interface address discovery not included
 **********************************************************************/
#ifdef HAVE_IF_ADDRS_DUMMY

std::vector<uhd::transport::if_addrs_t> uhd::transport::get_if_addrs(void)
{
    return std::vector<if_addrs_t>();
}

#endif /* HAVE_IF_ADDRS_DUMMY */