2023-03-06 14:42:04 +01:00
|
|
|
// SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
// author: Max Kellermann <max.kellermann@gmail.com>
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
#pragma once
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2020-07-23 17:35:09 +02:00
|
|
|
#include "SocketAddress.hxx" // IWYU pragma: export
|
2019-08-19 18:53:08 +02:00
|
|
|
#include "Features.hxx"
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2020-03-12 23:20:59 +01:00
|
|
|
#include <cassert>
|
2024-04-29 16:36:08 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_UN
|
2022-05-31 13:48:51 +02:00
|
|
|
#include <string_view>
|
2024-04-29 16:36:08 +02:00
|
|
|
#endif
|
2020-07-23 17:35:09 +02:00
|
|
|
|
2015-02-13 21:16:43 +01:00
|
|
|
/**
|
|
|
|
* An OO wrapper for struct sockaddr_storage.
|
|
|
|
*/
|
|
|
|
class StaticSocketAddress {
|
2017-08-10 12:17:35 +02:00
|
|
|
friend class SocketDescriptor;
|
|
|
|
|
2015-07-21 10:39:50 +02:00
|
|
|
public:
|
|
|
|
typedef SocketAddress::size_type size_type;
|
|
|
|
|
|
|
|
private:
|
|
|
|
size_type size;
|
2015-02-13 21:16:43 +01:00
|
|
|
struct sockaddr_storage address;
|
|
|
|
|
|
|
|
public:
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr StaticSocketAddress() noexcept = default;
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2023-08-30 10:28:17 +02:00
|
|
|
explicit StaticSocketAddress(SocketAddress src) noexcept {
|
|
|
|
*this = src;
|
|
|
|
}
|
|
|
|
|
2017-05-08 14:44:49 +02:00
|
|
|
StaticSocketAddress &operator=(SocketAddress other) noexcept;
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2020-01-30 16:51:18 +01:00
|
|
|
constexpr operator SocketAddress() const noexcept {
|
|
|
|
return SocketAddress(*this, size);
|
2015-02-13 21:16:43 +01:00
|
|
|
}
|
|
|
|
|
2020-01-30 16:51:18 +01:00
|
|
|
constexpr operator struct sockaddr *() noexcept {
|
|
|
|
return (struct sockaddr *)(void *)&address;
|
2017-08-10 12:40:58 +02:00
|
|
|
}
|
|
|
|
|
2020-01-30 16:51:18 +01:00
|
|
|
constexpr operator const struct sockaddr *() const noexcept {
|
|
|
|
return (const struct sockaddr *)(const void *)&address;
|
2017-08-10 12:40:58 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 21:35:43 +01:00
|
|
|
/**
|
|
|
|
* Cast the "sockaddr" pointer to a different address type,
|
|
|
|
* e.g. "sockaddr_in". This is only legal after checking
|
|
|
|
* GetFamily().
|
|
|
|
*/
|
|
|
|
template<typename T>
|
|
|
|
constexpr const T &CastTo() const noexcept {
|
|
|
|
/* cast through void to work around the bogus
|
|
|
|
alignment warning */
|
|
|
|
const void *q = reinterpret_cast<const void *>(&address);
|
|
|
|
return *reinterpret_cast<const T *>(q);
|
|
|
|
}
|
|
|
|
|
2017-06-03 21:33:44 +02:00
|
|
|
constexpr size_type GetCapacity() const noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
return sizeof(address);
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr size_type GetSize() const noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr void SetSize(size_type _size) noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
assert(_size > 0);
|
2015-07-21 10:39:50 +02:00
|
|
|
assert(size_t(_size) <= sizeof(address));
|
2015-02-13 21:16:43 +01:00
|
|
|
|
|
|
|
size = _size;
|
|
|
|
}
|
|
|
|
|
2017-08-10 12:24:59 +02:00
|
|
|
/**
|
|
|
|
* Set the size to the maximum value for this class.
|
|
|
|
*/
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr void SetMaxSize() {
|
2017-08-10 12:24:59 +02:00
|
|
|
SetSize(GetCapacity());
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr int GetFamily() const noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
return address.ss_family;
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr bool IsDefined() const noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
return GetFamily() != AF_UNSPEC;
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:52:05 +01:00
|
|
|
constexpr void Clear() noexcept {
|
2017-08-10 12:24:11 +02:00
|
|
|
size = sizeof(address.ss_family);
|
2015-02-13 21:16:43 +01:00
|
|
|
address.ss_family = AF_UNSPEC;
|
|
|
|
}
|
|
|
|
|
2024-01-05 22:46:45 +01:00
|
|
|
constexpr bool IsInet() const noexcept {
|
|
|
|
return GetFamily() == AF_INET || GetFamily() == AF_INET6;
|
|
|
|
}
|
|
|
|
|
2019-02-27 23:26:00 +01:00
|
|
|
#ifdef HAVE_UN
|
|
|
|
/**
|
|
|
|
* @see SocketAddress::GetLocalRaw()
|
|
|
|
*/
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2022-05-31 13:48:51 +02:00
|
|
|
std::string_view GetLocalRaw() const noexcept;
|
2024-04-29 16:33:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @see SocketAddress::GetLocalPath()
|
|
|
|
*/
|
|
|
|
[[nodiscard]] [[gnu::pure]]
|
|
|
|
const char *GetLocalPath() const noexcept {
|
|
|
|
return static_cast<const SocketAddress>(*this).GetLocalPath();
|
|
|
|
}
|
2019-02-27 23:26:00 +01:00
|
|
|
#endif
|
|
|
|
|
2017-05-17 12:03:45 +02:00
|
|
|
#ifdef HAVE_TCP
|
|
|
|
/**
|
|
|
|
* Extract the port number. Returns 0 if not applicable.
|
|
|
|
*/
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2017-06-04 13:09:11 +02:00
|
|
|
unsigned GetPort() const noexcept {
|
2017-05-17 12:03:45 +02:00
|
|
|
return ((SocketAddress)*this).GetPort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return true on success, false if this address cannot have
|
|
|
|
* a port number
|
|
|
|
*/
|
2017-06-04 13:09:11 +02:00
|
|
|
bool SetPort(unsigned port) noexcept;
|
2017-05-17 12:03:45 +02:00
|
|
|
#endif
|
|
|
|
|
2023-08-30 10:33:47 +02:00
|
|
|
[[gnu::pure]]
|
|
|
|
std::span<const std::byte> GetSteadyPart() const noexcept {
|
|
|
|
return SocketAddress{*this}.GetSteadyPart();
|
|
|
|
}
|
|
|
|
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2017-06-03 21:33:44 +02:00
|
|
|
bool operator==(SocketAddress other) const noexcept {
|
2015-07-22 10:54:17 +02:00
|
|
|
return (SocketAddress)*this == other;
|
|
|
|
}
|
2015-02-13 21:16:43 +01:00
|
|
|
|
2017-08-10 12:25:33 +02:00
|
|
|
bool operator!=(SocketAddress other) const noexcept {
|
2015-02-13 21:16:43 +01:00
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
};
|