libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
InterfaceAddress.hxx
1#pragma once
2
3// Linux
4#include <ifaddrs.h>
5#include <sys/types.h>
6
7// C++
8#include <optional>
9
10// cosmos
11#include <cosmos/BitMask.hxx>
12#include <cosmos/SysString.hxx>
13#include <cosmos/net/IPAddress.hxx>
14#include <cosmos/net/LinkLayerAddress.hxx>
15#include <cosmos/net/types.hxx>
16
17
18namespace cosmos {
19
21
29enum class InterfaceFlag : unsigned int {
30 UP = IFF_UP,
31 BROADCAST = IFF_BROADCAST,
32 DEBUG = IFF_DEBUG,
33 LOOPBACK = IFF_LOOPBACK,
34 POINTOPOINT = IFF_POINTOPOINT,
35 NOTRAILERS = IFF_NOTRAILERS,
36 RUNNING = IFF_RUNNING,
37 NOARP = IFF_NOARP,
38 PROMISC = IFF_PROMISC,
39 ALLMULTI = IFF_ALLMULTI,
40 MASTER = IFF_MASTER,
41 SLAVE = IFF_SLAVE,
42 MULTICAST = IFF_MULTICAST,
43 PORTSEL = IFF_PORTSEL,
44 AUTOMEDIA = IFF_AUTOMEDIA,
45 DYNAMIC = IFF_DYNAMIC,
46 LOWER_UP = IFF_LOWER_UP,
47 DORMANT = IFF_DORMANT,
48 ECHO = IFF_ECHO,
49};
50
52using InterfaceFlags = BitMask<InterfaceFlag>;
53
55
68 protected ::ifaddrs {
69
70public: // functions
71
72 // this type is only available via InterfaceAddressList, so disallow
73 // construction
74
75 InterfaceAddress() = delete;
76 InterfaceAddress(const InterfaceAddress&) = delete;
77
79 SysString ifname() const {
80 return ifa_name;
81 }
82
85 return InterfaceFlags{ifa_flags};
86 }
87
89
94 if (!hasAddress())
95 return SocketFamily::UNSPEC;
96
97 return SocketFamily{ifa_addr->sa_family};
98 }
99
101 bool hasAddress() const {
102 return ifa_addr != nullptr;
103 }
104
106 bool hasNetmask() const {
107 return ifa_netmask != nullptr;
108 }
109
111
115 bool hasBroadcastAddress() const {
116 if (!flags()[InterfaceFlag::BROADCAST])
117 return false;
118
119 return ifa_broadaddr != nullptr;
120 }
121
123 bool hasPointToPointDest() const {
124 if (!flags()[InterfaceFlag::POINTOPOINT])
125 return false;
126
127 return ifa_dstaddr != nullptr;
128 }
129
131 bool isIP4() const {
132 return family() == SocketFamily::INET;
133 }
134
136 bool isIP6() const {
137 return family() == SocketFamily::INET6;
138 }
139
141 bool isLinkLayer() const {
142 return family() == SocketFamily::PACKET;
143 }
144
146 std::optional<cosmos::IP4Address> addrAsIP4() const {
147 if (!isIP4())
148 return std::nullopt;
149
150 return cosmos::IP4Address{*(reinterpret_cast<sockaddr_in*>(ifa_addr))};
151 }
152
154 std::optional<cosmos::IP6Address> addrAsIP6() const {
155 if (!isIP6())
156 return std::nullopt;
157
158 return cosmos::IP6Address{*(reinterpret_cast<sockaddr_in6*>(ifa_addr))};
159 }
160
162 std::optional<cosmos::LinkLayerAddress> addrAsLLA() const {
163 if (!isLinkLayer())
164 return std::nullopt;
165
166 return cosmos::LinkLayerAddress{*(reinterpret_cast<sockaddr_ll*>(ifa_addr))};
167 }
168
170 std::optional<cosmos::IP4Address> netmaskAsIP4() const {
171 if (!hasNetmask() || SocketFamily{ifa_netmask->sa_family} != SocketFamily::INET)
172 return std::nullopt;
173
174 return cosmos::IP4Address{*(reinterpret_cast<sockaddr_in*>(ifa_netmask))};
175 }
176
178 std::optional<cosmos::IP6Address> netmaskAsIP6() const {
179 if (!hasNetmask() || SocketFamily{ifa_netmask->sa_family} != SocketFamily::INET6)
180 return std::nullopt;
181
182 return cosmos::IP6Address{*(reinterpret_cast<sockaddr_in6*>(ifa_netmask))};
183 }
184
186 std::optional<cosmos::IP4Address> broadcastAsIP4() const {
187 if (!hasBroadcastAddress() || SocketFamily{ifa_broadaddr->sa_family} != SocketFamily::INET)
188 return std::nullopt;
189
190 return cosmos::IP4Address{*(reinterpret_cast<sockaddr_in*>(ifa_broadaddr))};
191 }
192
194 std::optional<cosmos::IP4Address> pointToPointAsIP4() const {
195 if (!hasPointToPointDest() || SocketFamily{ifa_broadaddr->sa_family} != SocketFamily::INET)
196 return std::nullopt;
197
198 return cosmos::IP4Address{*(reinterpret_cast<sockaddr_in*>(ifa_dstaddr))};
199 }
200};
201
202} // end ns
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
A 32-bit IPv4 address and 16 bit port number for use with SocketFamily::INET sockets.
A 128 bit IPv6 address and 16-bit port number plus some IPv6 specific extra fields.
A single network interface address.
std::optional< cosmos::IP4Address > broadcastAsIP4() const
If an IPv4 broadcast address is available, return it.
std::optional< cosmos::IP4Address > pointToPointAsIP4() const
If an IPv4 point-to-point destination is available, return it.
std::optional< cosmos::IP6Address > addrAsIP6() const
If this is an IPv6 address, return it.
SocketFamily family() const
Returns the SocketFamily this address is about.
std::optional< cosmos::LinkLayerAddress > addrAsLLA() const
If this is a link layer address, return it.
bool hasNetmask() const
Returns whether a netmask address is available in this entry.
bool hasPointToPointDest() const
Returns whether a point-to-point destination is available in this entry.
bool hasAddress() const
Returns whether an address is available in this entry.
InterfaceFlags flags() const
Returns the current interface status flags.
std::optional< cosmos::IP4Address > addrAsIP4() const
If this is an IPv4 address, return it.
std::optional< cosmos::IP6Address > netmaskAsIP6() const
If an IPv6 netmask is available, return it.
bool isIP4() const
Returns whether the interface address is an IPv4 address.
bool hasBroadcastAddress() const
Returns whether a broadcast address is available in this entry.
SysString ifname() const
Returns the unique string that identifies the network device that this address belongs to.
bool isLinkLayer() const
Returns whether the interface address is a LinkLayerAddress.
std::optional< cosmos::IP4Address > netmaskAsIP4() const
If an IPv4 netmask is available, return it.
bool isIP6() const
Returns whether the interface address is an IPv6 address.
A link layer (network layer 2) socket address.
SocketFamily
A socket's family setting.
Definition types.hxx:37
Wrapper type around a C-style string for use with system APIs.
Definition SysString.hxx:33