blob: c299ed0706c14b1bd650f40711ee2529acb275a3 [file] [log] [blame]
William A. Kennington IIIeac9d472020-08-03 13:57:14 -07001#include <stdplus/exception.hpp>
2#include <stdplus/fd/ops.hpp>
Patrick Williamsd1984dd2023-05-10 16:12:44 -05003
William A. Kennington III6417c632023-07-17 02:56:52 -07004#include <format>
William A. Kennington IIIeac9d472020-08-03 13:57:14 -07005#include <utility>
6
7namespace stdplus
8{
9namespace fd
10{
11namespace detail
12{
13
14template <typename Fun, typename Byte, typename... Args>
Patrick Williams68975b92022-04-27 16:00:26 -050015static void opExact(const char* name, Fun&& fun, Fd& fd, std::span<Byte> data,
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070016 Args&&... args)
17{
William A. Kennington III93ae9702023-12-22 16:41:17 -080018 std::size_t total = 0;
19 try
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070020 {
William A. Kennington III93ae9702023-12-22 16:41:17 -080021 while (total < data.size())
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070022 {
William A. Kennington III93ae9702023-12-22 16:41:17 -080023 auto r = (fd.*fun)(data.subspan(total),
24 std::forward<Args>(args)...);
25 if (r.size() == 0)
26 {
27 throw exception::WouldBlock(
28 std::format("{} missing", name, data.size()));
29 }
30 total += r.size();
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070031 }
William A. Kennington III93ae9702023-12-22 16:41:17 -080032 }
33 catch (const std::system_error&)
34 {
35 if (total != 0)
36 {
37 throw exception::Incomplete(
38 std::format("{} is {}B/{}B", name, total, data.size()));
39 }
40 throw;
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070041 }
42}
43
Patrick Williams68975b92022-04-27 16:00:26 -050044void readExact(Fd& fd, std::span<std::byte> data)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070045{
46 opExact("readExact", &Fd::read, fd, data);
47}
48
Patrick Williams68975b92022-04-27 16:00:26 -050049void recvExact(Fd& fd, std::span<std::byte> data, RecvFlags flags)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070050{
51 opExact("recvExact", &Fd::recv, fd, data, flags);
52}
53
Patrick Williams68975b92022-04-27 16:00:26 -050054void writeExact(Fd& fd, std::span<const std::byte> data)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070055{
56 opExact("writeExact", &Fd::write, fd, data);
57}
58
Patrick Williams68975b92022-04-27 16:00:26 -050059void sendExact(Fd& fd, std::span<const std::byte> data, SendFlags flags)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070060{
61 opExact("sendExact", &Fd::send, fd, data, flags);
62}
63
William A. Kennington III4edcc6e2023-12-19 17:24:54 -080064void sendtoExact(Fd& fd, std::span<const std::byte> data, SendFlags flags,
65 std::span<const std::byte> addr)
66{
67 auto r = fd.sendto(data, flags, addr);
68 if (r.size() == 0)
69 {
70 throw exception::WouldBlock("sendto");
71 }
72 if (r.size() < data.size())
73 {
74 throw exception::Incomplete("sendto");
75 }
76}
77
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070078template <typename Fun, typename Byte, typename... Args>
Patrick Williams68975b92022-04-27 16:00:26 -050079static std::span<Byte> opAligned(const char* name, Fun&& fun, Fd& fd,
80 size_t align, std::span<Byte> data,
81 Args&&... args)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070082{
William A. Kennington IIIb88c4572023-12-22 16:43:42 -080083 std::size_t total = 0;
84 try
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070085 {
William A. Kennington IIIb88c4572023-12-22 16:43:42 -080086 do
William A. Kennington IIIeac9d472020-08-03 13:57:14 -070087 {
William A. Kennington IIIb88c4572023-12-22 16:43:42 -080088 auto r = (fd.*fun)(data.subspan(total),
89 std::forward<Args>(args)...);
90 if (total != 0 && r.size() == 0)
91 {
92 throw exception::Incomplete(
93 std::format("{} is {}B/{}B", name, total % align, align));
94 }
95 total += r.size();
96 } while (total % align != 0);
97 }
98 catch (const std::system_error&)
99 {
100 if (total % align)
101 {
102 throw exception::Incomplete(
103 std::format("{} is {}B/{}B", name, total % align, align));
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700104 }
William A. Kennington IIIb88c4572023-12-22 16:43:42 -0800105 throw;
106 }
107 return std::span<Byte>(data.data(), total);
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700108}
109
Patrick Williams68975b92022-04-27 16:00:26 -0500110std::span<std::byte> readAligned(Fd& fd, size_t align, std::span<std::byte> buf)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700111{
112 return opAligned("readAligned", &Fd::read, fd, align, buf);
113}
114
Patrick Williams68975b92022-04-27 16:00:26 -0500115std::span<std::byte> recvAligned(Fd& fd, size_t align, std::span<std::byte> buf,
116 RecvFlags flags)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700117{
118 return opAligned("recvAligned", &Fd::recv, fd, align, buf, flags);
119}
120
Patrick Williams68975b92022-04-27 16:00:26 -0500121std::span<const std::byte> writeAligned(Fd& fd, size_t align,
122 std::span<const std::byte> data)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700123{
124 return opAligned("writeAligned", &Fd::write, fd, align, data);
125}
126
Patrick Williams68975b92022-04-27 16:00:26 -0500127std::span<const std::byte> sendAligned(Fd& fd, size_t align,
128 std::span<const std::byte> data,
129 SendFlags flags)
William A. Kennington IIIeac9d472020-08-03 13:57:14 -0700130{
131 return opAligned("sendAligned", &Fd::send, fd, align, data, flags);
132}
133
134} // namespace detail
135} // namespace fd
136} // namespace stdplus