libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
SocketError.hxx
1#pragma once
2
3// C++
4#include <cstdint>
5#include <optional>
6#include <type_traits>
7#include <utility>
8
9// Linux
10#include <sys/time.h>
11#include <linux/errqueue.h>
12
13// Cosmos
14#include <cosmos/error/errno.hxx>
15#include <cosmos/net/IPAddress.hxx>
16#include <cosmos/net/traits.hxx>
17#include <cosmos/net/types.hxx>
18
19namespace cosmos {
20
22
34template <SocketFamily FAMILY>
36 protected sock_extended_err {
37public: // types
38
39 using IPAddress = typename FamilyTraits<FAMILY>::Address;
40
42 enum class Origin : uint8_t {
43 NONE = SO_EE_ORIGIN_NONE,
45 LOCAL = SO_EE_ORIGIN_LOCAL,
47 ICMP = SO_EE_ORIGIN_ICMP,
49 ICMP6 = SO_EE_ORIGIN_ICMP6,
50 TXSTATUS = SO_EE_ORIGIN_TXSTATUS,
52 ZEROCOPY = SO_EE_ORIGIN_ZEROCOPY,
53 TXTIME = SO_EE_ORIGIN_TXTIME
54 };
55
57 enum class ZeroCopyCode : uint8_t {
59 ZEROCOPY_COPIED = SO_EE_CODE_ZEROCOPY_COPIED,
60 };
61
63 enum class TxTimeCode : uint8_t {
64 TXTIME_INVALID_PARAM = SO_EE_CODE_TXTIME_INVALID_PARAM,
65 TXTIME_MISSED = SO_EE_CODE_TXTIME_MISSED
66 };
67
68public: // functions
69
71 Origin origin() const {
72 return Origin{this->ee_origin};
73 }
74
76 Errno errnum() const {
77 // ee_errno is unsigned, while errno is signed
78 return Errno{static_cast<std::underlying_type<Errno>::type>(this->ee_errno)};
79 }
80
82 std::optional<uint32_t> discoveredMTU() const {
83 if (errnum() == Errno::MSG_TOO_LARGE) {
84 return this->ee_info;
85 }
86
87 return std::nullopt;
88 }
89
90 std::optional<ZeroCopyCode> zeroCopyCode() const {
91 if (origin() == Origin::ZEROCOPY) {
92 return ZeroCopyCode{this->ee_code};
93 }
94
95 return std::nullopt;
96 }
97
98 std::optional<TxTimeCode> txTimeCode() const {
99 if (origin() == Origin::TXTIME) {
100 return TxTimeCode{this->ee_code};
101 }
102
103 return std::nullopt;
104 }
105
107
111 std::optional<std::pair<uint32_t, uint32_t>> zeroCopyRange() const {
112 if (origin() == Origin::ZEROCOPY) {
113 return std::make_pair(this->ee_info, this->ee_data);
114 }
115
116 return std::nullopt;
117 }
118
119 bool originIsICMP() const {
120 return origin() == Origin::ICMP || origin() == Origin::ICMP6;
121 }
122
123 std::optional<uint8_t> icmpType() const {
124 if (originIsICMP()) {
125 return this->ee_type;
126 }
127
128 return std::nullopt;
129 }
130
131 std::optional<uint8_t> icmpCode() const {
132 if (originIsICMP()) {
133 return this->ee_code;
134 }
135
136 return std::nullopt;
137 }
138
139 SocketFamily offenderAddressFamily() const {
140 return SocketFamily(offenderAddr()->sa_family);
141 }
142
144
149 bool hasOffenderAddress() const {
150 return offenderAddressFamily() != SocketFamily::UNSPEC;
151 }
152
153 std::optional<IPAddress> offenderAddress() const {
154 if (offenderAddressFamily() != FAMILY)
155 return std::nullopt;
156
157 using RawAddr = typename FamilyTraits<FAMILY>::RawAddr;
158 return IPAddress{*reinterpret_cast<const RawAddr*>(offenderAddr())};
159 }
160
161protected: // functions
162
163 const struct sockaddr* offenderAddr() const {
164 // this is piggyback data found after the end of the struct
165 // sock_extended_err. It is only valid if sa_family !=
166 // AF_UNSPEC.
167 return SO_EE_OFFENDER(this);
168 }
169};
170
171using IP4SocketError = SocketErrorT<SocketFamily::INET>;
172using IP6SocketError = SocketErrorT<SocketFamily::INET6>;
173
174} // end ns
Wrapper for socket extended errors ancillary message of types IP4Message::RECVERR and IP6Message::REC...
std::optional< std::pair< uint32_t, uint32_t > > zeroCopyRange() const
Return the copied ranges for zerocopy status reports.
std::optional< uint32_t > discoveredMTU() const
If errnum() is Errno::MSG_TOO_LARGE then this returns the currently known MTU.
Origin
This defines where the extended error originated.
@ ZEROCOPY
Status report for zerocopy operation (see Linux kernel documentation networking/msg_zerocopy....
@ LOCAL
The local networking stack detected an error.
@ ICMP
An ICMPv4 error was reported.
@ ICMP6
An ICMPv6 error was reported.
bool hasOffenderAddress() const
Check whether the offender IP address is available.
Origin origin() const
The origin defines how the rest of the error data is interpreted.
TxTimeCode
Code definitions for Origin::TXTIME.
ZeroCopyCode
Code definitions for Origin::ZEROCOPY.
@ ZEROCOPY_COPIED
No zerocopy was performed, the kernel performed a copy.
Errno errnum() const
The error code is always available, but may be Errno::NO_ERROR.
Errno
Strong enum type representing errno error constants.
Definition errno.hxx:29
SocketFamily
A socket's family setting.
Definition types.hxx:37