libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
Socket.cxx
1// cosmos
2#include <cosmos/error/ApiError.hxx>
3#include <cosmos/error/RuntimeError.hxx>
4#include <cosmos/net/message_header.hxx>
5#include <cosmos/net/SocketAddress.hxx>
6#include <cosmos/net/Socket.hxx>
7#include <cosmos/utils.hxx>
8
9namespace cosmos {
10
12 const SocketFamily family,
13 const SocketType type,
14 const SocketFlags flags,
15 const SocketProtocol protocol) {
16 auto fd = ::socket(
17 to_integral(family),
18 to_integral(type) | flags.raw(),
19 to_integral(protocol));
20 m_fd.setFD(FileNum{fd});
21
22 if (!m_fd.valid()) {
23 cosmos_throw(ApiError("socket()"));
24 }
25}
26
27void Socket::bind(const SocketAddress &addr) {
28 if (::bind(to_integral(m_fd.raw()), addr.basePtr(), addr.size()) != 0) {
29 cosmos_throw(ApiError("bind()"));
30 }
31}
32
33void Socket::connect(const SocketAddress &addr) {
34 if (::connect(to_integral(m_fd.raw()), addr.basePtr(), addr.size()) != 0) {
35 cosmos_throw(ApiError("connect()"));
36 }
37}
38
39void Socket::shutdown(const Direction dir) {
40 if (::shutdown(to_integral(m_fd.raw()), to_integral(dir)) != 0) {
41 cosmos_throw (ApiError("shutdown()"));
42 }
43}
44
45void Socket::listen(const size_t backlog) {
46 if (::listen(to_integral(m_fd.raw()), backlog) != 0) {
47 cosmos_throw (ApiError("listen()"));
48 }
49}
50
52 socklen_t addrlen = addr ? addr->maxSize() : 0;
53 auto res = ::accept(to_integral(m_fd.raw()), addr ? addr->basePtr() : nullptr, addr ? &addrlen : nullptr);
54
55 if (res == -1) {
56 cosmos_throw (ApiError("accept()"));
57 }
58
59 if (addr) {
60 addr->update(addrlen);
61 }
62
63 return FileDescriptor{FileNum{res}};
64}
65
66size_t Socket::send(const void *buf, size_t length, const MessageFlags flags) {
67 const auto res = ::send(to_integral(m_fd.raw()), buf, length, flags.raw());
68 if (res < 0) {
69 cosmos_throw(ApiError("send()"));
70 }
71
72 return static_cast<size_t>(res);
73}
74
75size_t Socket::sendTo(const void *buf, size_t length, const SocketAddress &addr, const MessageFlags flags) {
76 const auto res = ::sendto(to_integral(m_fd.raw()), buf, length, flags.raw(), addr.basePtr(), addr.size());
77 if (res < 0) {
78 cosmos_throw(ApiError("sendto()"));
79 }
80
81 return static_cast<size_t>(res);
82}
83
84size_t Socket::receive(void *buf, size_t length, const MessageFlags flags) {
85 const auto res = ::recv(to_integral(m_fd.raw()), buf, length, flags.raw());
86 if (res < 0) {
87 cosmos_throw(ApiError("recv()"));
88 }
89
90 return static_cast<size_t>(res);
91}
92
93std::pair<size_t, Socket::AddressFilledIn>
94Socket::receiveFrom(void *buf, size_t length, SocketAddress &addr, const MessageFlags flags) {
95 socklen_t addrlen = addr.maxSize();
96 const auto res = ::recvfrom(to_integral(m_fd.raw()), buf, length, flags.raw(), addr.basePtr(), &addrlen);
97 if (res < 0) {
98 cosmos_throw(ApiError("recvfrom()"));
99 }
100
101 const AddressFilledIn filled_in{addrlen != 0};
102
103 // when no sender's address is available addrlen will be zero.
104 if (filled_in) {
105 addr.update(addrlen);
106 }
107
108 return {static_cast<size_t>(res), filled_in};
109}
110
112 socklen_t size = addr.maxSize();
113 auto base = addr.basePtr();
114 if (::getsockname(to_integral(m_fd.raw()), base, &size) != 0) {
115 cosmos_throw(ApiError("getsockname()"));
116 }
117
118 if (SocketFamily{base->sa_family} != addr.family()) {
119 addr.clear();
120 cosmos_throw(RuntimeError("getsockname: wrong type of SocketAddress was passed"));
121 }
122
123 addr.update(size);
124}
125
127 header.prepareSend(addr);
128
129 const auto res = ::sendmsg(
130 to_integral(m_fd.raw()),
131 header.rawHeader(),
132 header.ioFlags().raw());
133
134 if (res < 0) {
135 cosmos_throw(ApiError("sendmsg()"));
136 }
137
138 // NOTE: if a control message is configured in the header, then even
139 // on partial writes the control message will always be transmitted.
140 // The result will contain the bytes of the control message, only the
141 // actual message payload.
142 header.postSend(res);
143}
144
146 header.prepareReceive(addr);
147
148 const auto res = ::recvmsg(
149 to_integral(m_fd.raw()),
150 header.rawHeader(),
151 header.ioFlags().raw());
152
153 if (res < 0) {
154 cosmos_throw(ApiError("recvmsg()"));
155 }
156
157 auto ret = Socket::AddressFilledIn{false};
158
159 if (addr) {
160 if (const auto namelen = header.rawHeader()->msg_namelen; namelen != 0) {
161 ret = Socket::AddressFilledIn{true};
162 addr->update(namelen);
163 }
164 }
165
166 header.postReceive(res);
167
168 return ret;
169}
170
171} // end ns
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
EnumBaseType raw() const
Returns the raw bitfield integer.
Definition BitMask.hxx:56
FileDescriptor fd() const
Allows access to the underlying fd with const semantics.
Definition FileBase.hxx:74
Thin Wrapper around OS file descriptors.
FileNum raw() const
Returns the primitive file descriptor contained in the object.
void setFD(const FileNum fd)
Assigns a new primitive file descriptor to the object.
bool valid() const
Returns whether currently a valid file descriptor number is assigned.
MessageFlags ioFlags() const
Returns the currently set MessageFlags for send/receive.
Strong template type to wrap boolean values in a named type.
Definition utils.hxx:50
Wrapper for struct msghdr for receiving message via Socket::receiveMessage().
void prepareReceive(SocketAddress *addr)
Prepare a recvmsg() operation using the given optional source address storage.
Exception type for generic runtime errors.
Wrapper for struct msghdr for sending messages via Socket::sendMessage().
void postSend(size_t sent)
Perform any cleanup or bookkeeping after a successful sendmsg() operation.
const struct msghdr * rawHeader() const
Return a pointer to the raw struct msghdr for passing to the sendmsg() system call.
void prepareSend(const SocketAddress *addr)
Prepare a sendmsg() operation using the given optional target address.
Base class for all types of socket addresses.
void clear()
Clears the complete address structure.
virtual void update(size_t new_length)
Update the address structure after it has been filled in by the kernel.
virtual size_t maxSize() const
Returns the maximum number of bytes the socket address can hold.
virtual SocketFamily family() const =0
Returns the concrete SocketFamily for the implementation address type.
virtual size_t size() const =0
Returns the size of the socket address in bytes found at basePtr().
virtual sockaddr * basePtr()=0
Returns a mutable pointer to the sockaddr* base structure.
void shutdown(const Direction dir)
Shutdown part or all of the connection on protocol level.
Definition Socket.cxx:39
void sendMessage(SendMessageHeader &header, const SocketAddress *addr=nullptr)
Sends a message over the socket using extended SendMessageHeader data.
Definition Socket.cxx:126
std::pair< size_t, AddressFilledIn > receiveFrom(void *buf, size_t length, SocketAddress &addr, const MessageFlags flags=MessageFlags{})
Receive a packet, filling in the sender's address.
Definition Socket.cxx:94
void connect(const SocketAddress &addr)
Establish a new connection using the given destination address.
Definition Socket.cxx:33
Socket(const SocketFamily family, const SocketType type, const SocketFlags flags=SocketFlags{SocketFlag::CLOEXEC}, const SocketProtocol protocol=SocketProtocol::DEFAULT)
Creates a new socket using the given properties.
Definition Socket.cxx:11
void listen(const size_t backlog)
Enter into a passive listen state, allowing new connections.
Definition Socket.cxx:45
void bind(const SocketAddress &addr)
Bind the socket to the given local address.
Definition Socket.cxx:27
Direction
Type used in Socket::shutdown().
Definition Socket.hxx:43
AddressFilledIn receiveMessage(ReceiveMessageHeader &header, SocketAddress *addr=nullptr)
Receives a message from the socket using extended ReceiveMessageHeader data.
Definition Socket.cxx:145
void getSockName(SocketAddress &addr)
Returns the current address that the socket is bound to, if any.
Definition Socket.cxx:111
FileDescriptor accept(SocketAddress *addr)
Accept a new connection on the socket.
Definition Socket.cxx:51
NamedBool< struct addr_filled_in_t, false > AddressFilledIn
Boolean flag used in receiveFrom() to signify if a peer address could be provided.
Definition Socket.hxx:50
size_t sendTo(const void *buf, size_t length, const SocketAddress &addr, const MessageFlags flags=MessageFlags{})
Send a packet to a specific destination address.
Definition Socket.cxx:75
size_t receive(void *buf, size_t length, const MessageFlags flags=MessageFlags{})
Receive data from the socket, using specific receive flags.
Definition Socket.cxx:84
size_t send(const void *buf, size_t length, const MessageFlags flags=MessageFlags{})
Send the given data over the socket, using specific send flags.
Definition Socket.cxx:66
FileNum
Primitive file descriptor.
Definition types.hxx:32
SocketProtocol
Specific protocol to use on a socket.
Definition types.hxx:71
SocketFamily
A socket's family setting.
Definition types.hxx:37
SocketType
A socket's type setting.
Definition types.hxx:52