libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
IPAddress.hxx
1#pragma once
2
3// Linux
4#include <netdb.h>
5
6// C++
7#include <cstring>
8#include <string>
9
10// cosmos
11#include <cosmos/BitMask.hxx>
12#include <cosmos/SysString.hxx>
13#include <cosmos/dso_export.h>
14#include <cosmos/net/SocketAddress.hxx>
15#include <cosmos/net/byte_order.hxx>
16#include <cosmos/utils.hxx>
17
18namespace cosmos {
19
21
25class COSMOS_API IPAddressBase :
26 public SocketAddress {
27public: // types
28
30 enum class NameInfoFlag : int {
32 NAME_REQUIRED = NI_NAMEREQD,
34 DGRAM = NI_DGRAM,
36 NO_FQDN = NI_NOFQDN,
38 NUMERIC_HOST = NI_NUMERICHOST,
40 NUMERIC_SERVICE = NI_NUMERICSERV,
42 IDN = NI_IDN,
43#if 0 /* these are declared as deprecated */
44 IDN_ALLOW_UNASSIGNED = NI_IDN_ALLOW_UNASSIGNED,
45 IDN_USE_STD3_ASCII_RULES = NI_IDN_USE_STD3_ASCII_RULES,
46#endif
47 };
48
51
52public: // functions
53
54 bool isV4() const { return this->family() == SocketFamily::INET; }
55 bool isV6() const { return this->family() == SocketFamily::INET6; }
56
58 std::string ipAsString() const;
60 void setIpFromString(const SysString str);
61
63
71 void getNameInfo(std::string &host, std::string &service, const NameInfoFlags flags = {});
72
74
80 std::string getHostInfo(const NameInfoFlags flags = {});
81
83
87 std::string getServiceInfo(const NameInfoFlags flags = {});
88
89protected: // functions
90
92 void* ipAddrPtr();
94 const void* ipAddrPtr() const;
95
96 void getNameInfo(std::string *host, std::string *service, const NameInfoFlags flags);
97};
98
101 public IPAddressBase {
102public: // types
103
104 static constexpr inline SocketFamily FAMILY = SocketFamily::INET;
105
106public: // functions
107
108 IP4Address() {
109 clear();
110 }
111
112 explicit IP4Address(const sockaddr_in &raw) {
113 m_addr = raw;
114 }
115
116 explicit IP4Address(const IP4RawAddress addr, const IPPort port = IPPort{0}) {
117 setFamily();
118 setAddr(addr);
119 setPort(port);
120 }
121
122 explicit IP4Address(const SysString ip, const IPPort port = IPPort{0}) {
123 setFamily();
124 setIpFromString(ip);
125 setPort(port);
126 }
127
128 SocketFamily family() const override {
129 return FAMILY;
130 }
131
132 size_t size() const override {
133 return sizeof(m_addr);
134 }
135
136 net::NetInt16 port() const { return net::NetInt16{net::RawNetInt16{m_addr.sin_port}}; }
137 void setPort(const net::NetInt16 port) { m_addr.sin_port = to_integral(port.raw()); }
138
139 IP4RawAddress addr() const { return IP4RawAddress{net::RawNetInt32{m_addr.sin_addr.s_addr}}; }
140 void setAddr(const IP4RawAddress addr) { m_addr.sin_addr.s_addr = to_integral(addr.raw()); }
141
142 bool operator==(const IP4Address &other) const {
143 return addr() == other.addr() && port() == other.port();
144 }
145
146 bool operator!=(const IP4Address &other) const {
147 return !(*this == other);
148 }
149
150protected: // functions
151
152 sockaddr* basePtr() override {
153 return reinterpret_cast<sockaddr*>(&m_addr);
154 }
155
156 const sockaddr* basePtr() const override {
157 return reinterpret_cast<const sockaddr*>(&m_addr);
158 }
159
160 void setFamily() {
161 m_addr.sin_family = to_integral(family());
162 }
163
164protected: // data
165
166 sockaddr_in m_addr;
167};
168
171 public IPAddressBase {
172public: // data
173
174 static constexpr inline SocketFamily FAMILY = SocketFamily::INET6;
175
176public: // functions
177
178 IP6Address() {
179 clear();
180 }
181
182 explicit IP6Address(const sockaddr_in6 &raw) {
183 m_addr = raw;
184 }
185
186 explicit IP6Address(const IP6RawAddress &addr, const IPPort port = IPPort{0}) {
187 clear();
188 setAddr(addr);
189 setPort(port);
190 }
191
192 SocketFamily family() const override {
193 return FAMILY;
194 }
195
196 size_t size() const override {
197 return sizeof(m_addr);
198 }
199
200 IPPort port() const { return IPPort{net::RawNetInt16{m_addr.sin6_port}}; }
201 void setPort(const IPPort port) { m_addr.sin6_port = to_integral(port.raw()); }
202
203 IP6RawAddress addr() const {
204 IP6RawAddress ret;
205 std::memcpy(ret.data(), m_addr.sin6_addr.s6_addr, ret.size());
206 return ret;
207 }
208
209 void setAddr(const IP6RawAddress &addr) {
210 std::memcpy(m_addr.sin6_addr.s6_addr, addr.begin(), addr.size());
211 }
212
214
220 // the type for the scope_id and index are inconsistent
221 // overly large scope_ids should not occur in practice though,
222 // so we shouldn't run into trouble here.
223 return InterfaceIndex{static_cast<int>(m_addr.sin6_scope_id)};
224 }
225
227 void setScopeID(const InterfaceIndex index) {
228 m_addr.sin6_scope_id = to_integral(index);
229 }
230
232
237 uint32_t getFlowInfo() const {
238 return m_addr.sin6_flowinfo;
239 }
240
241 void setFlowInfo(const uint32_t flowinfo) {
242 m_addr.sin6_flowinfo = flowinfo;
243 }
244
245 bool operator==(const IP6Address &other) const {
246 const auto cmp_res = std::memcmp(m_addr.sin6_addr.s6_addr,
247 other.m_addr.sin6_addr.s6_addr,
248 sizeof(m_addr.sin6_addr.s6_addr));
249 return cmp_res == 0 && port() == other.port();
250 }
251
252 bool operator!=(const IP6Address &other) const {
253 return !(*this == other);
254 }
255
256protected: // functions
257
258 sockaddr* basePtr() override {
259 return reinterpret_cast<sockaddr*>(&m_addr);
260 }
261
262 const sockaddr* basePtr() const override {
263 return reinterpret_cast<const sockaddr*>(&m_addr);
264 }
265
266 void setFamily() {
267 m_addr.sin6_family = to_integral(family());
268 }
269
270protected: // data
271
272 sockaddr_in6 m_addr;
273};
274
275} // 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.
sockaddr * basePtr() override
Returns a mutable pointer to the sockaddr* base structure.
const sockaddr * basePtr() const override
Returns a const pointer to the sockaddr* base structure.
size_t size() const override
Returns the size of the socket address in bytes found at basePtr().
SocketFamily family() const override
Returns the concrete SocketFamily for the implementation address type.
A 32-bit IPv4 binary address in network byte order.
Definition types.hxx:130
A 128 bit IPv6 address and 16-bit port number plus some IPv6 specific extra fields.
const sockaddr * basePtr() const override
Returns a const pointer to the sockaddr* base structure.
sockaddr * basePtr() override
Returns a mutable pointer to the sockaddr* base structure.
SocketFamily family() const override
Returns the concrete SocketFamily for the implementation address type.
InterfaceIndex getScopeID() const
Returns the currently set scope ID.
void setScopeID(const InterfaceIndex index)
Set a new scope ID interface index.
size_t size() const override
Returns the size of the socket address in bytes found at basePtr().
uint32_t getFlowInfo() const
Returns the IPv6 flow info identifier.
Base class for both IPv4 and IPv6 addresses.
Definition IPAddress.hxx:26
NameInfoFlag
Flags used with the getNameInfo() function.
Definition IPAddress.hxx:30
void setIpFromString(const SysString str)
Sets the binary IP address from the given string.
Definition IPAddress.cxx:43
A 16-bit IP port in network byte order.
Definition types.hxx:125
Base class for all types of socket addresses.
void clear()
Clears the complete address structure.
An endianness aware unsigned integer.
InterfaceIndex
A network device interface index.
Definition types.hxx:119
SocketFamily
A socket's family setting.
Definition types.hxx:37
@ DGRAM
connection-less, unreliable, unordered with length limitation, keeps message boundaries.
A 128-bit IPv6 address.
Definition types.hxx:148
Wrapper type around a C-style string for use with system APIs.
Definition SysString.hxx:33