| // Copyright 2017 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <lib/fitx/result.h> |
| #include <lib/stdcompat/span.h> |
| #include <lib/zx/socket.h> |
| #include <lib/zxio/cpp/create_with_type.h> |
| #include <lib/zxio/cpp/inception.h> |
| #include <lib/zxio/cpp/vector.h> |
| #include <lib/zxio/null.h> |
| #include <net/if.h> |
| #include <netinet/icmp6.h> |
| #include <netinet/if_ether.h> |
| #include <netinet/in.h> |
| #include <netinet/tcp.h> |
| #include <netinet/udp.h> |
| #include <poll.h> |
| #include <sys/ioctl.h> |
| |
| #include <algorithm> |
| #include <optional> |
| #include <type_traits> |
| #include <vector> |
| |
| #include <netpacket/packet.h> |
| #include <safemath/safe_conversions.h> |
| |
| #include "fdio_unistd.h" |
| #include "zxio.h" |
| |
| namespace fsocket = fuchsia_posix_socket; |
| namespace frawsocket = fuchsia_posix_socket_raw; |
| namespace fpacketsocket = fuchsia_posix_socket_packet; |
| namespace fnet = fuchsia_net; |
| |
| /* Socket class hierarchy |
| * |
| * Wrapper structs for supported FIDL protocols that encapsulate associated |
| * types and specialized per-protocol logic. |
| * |
| * +-------------------------+ +---------------------+ |
| * | struct StreamSocket | | struct RawSocket | |
| * | fsocket::StreamSocket | | frawsocket:Socket | |
| * +-------------------------+ +---------------------+ |
| * +-------------------------+ +-------------------------------------+ |
| * | struct PacketSocket | | struct SynchronousDatagramSocket | |
| * | fpacketsocket::Socket | | fsocket:SynchronousDatagramSocket | |
| * +-------------------------+ +-------------------------------------+ |
| * |
| * Utility classes constructed on-the-fly for common socket operations, enabled |
| * for relevant FIDL wrappers: |
| * |
| * +-------------------------+ +-------------------------+ |
| * | class BaseSocket | | class NetworkSocket | |
| * | | | | |
| * | Enabled: | | Enabled: | |
| * | RawSocket | | RawSocket | |
| * | SyncDgramSocket | | SyncDgramSocket | |
| * | StreamSocket + -------> | StreamSocket | |
| * | PacketSocket | | | |
| * | | | Implements: | |
| * | Implements: | | Operations for network | |
| * | Operations for | | sockets | |
| * | all socket types | | | |
| * +-------------------------+ +-------------------------+ |
| * |
| * Stateful class hierarchy for wrapping zircon primitives, enabled for |
| * relevant FIDL wrappers: |
| * |
| * +-----------------------------+ |
| * | network_socket_with_event | +-----------------+ |
| * +---------------+ | | | stream_socket | |
| * | packet_socket | | Enabled: | | | |
| * | | | RawSocket | | Enabled: | |
| * | Enabled: | | SyncDgramSocket | | StreamSocket | |
| * | PacketSocket | | | | | |
| * | | | | | Implements: | |
| * | Implements: | | Implements: Template | | Overrides for | |
| * | Overrides for | | for instantiating | | sockets using | |
| * | packet | | network sockets that | | zx::socket as | |
| * | sockets | | use FIDL over channel | | a data plane | |
| * | (AF_PACKET) | | (SOCK_RAW, SOCK_DGRAM) | | (SOCK_STREAM) | |
| * +---------------+ +-----------------------------+ +-----------------+ |
| * ^ ^ ^ ^ |
| * | | | | |
| * | | | | |
| * +-----------+------------+ +-------------+ | |
| * | | | |
| * | | | |
| * +-----------+-----------+ +-------+------+--------+ |
| * | socket_with_event | | network_socket | |
| * | | | | |
| * | Enabled: | | Enabled: | |
| * | PacketSocket | | RawSocket | |
| * | RawSocket | | SyncDgramSocket | |
| * | SyncDgramSocket | | Streamsocket | |
| * | | <---------+---------> | | |
| * | Implements: Overrides | | | Implements: Overrides | |
| * | for sockets using | | | for network layer | |
| * | FIDL over channel | | | sockets | |
| * | as a data plane | | | | |
| * +-----------------------+ | +-----------------------+ |
| * | |
| * +--------------+-------------+ |
| * | base_socket | |
| * | | |
| * | Enabled: All | |
| * | | |
| * | Implements: Overrides for | |
| * | all socket types | |
| * +----------------------------+ |
| * ^ |
| * | |
| * | |
| * +----------+-----------+ |
| * | zxio | |
| * | | |
| * | Implements: POSIX | |
| * | interface + behavior | |
| * | for generic fds | |
| * +----------------------+ |
| */ |
| |
| namespace { |
| |
| // A helper structure to keep a socket address and the variants allocations on the stack. |
| struct SocketAddress { |
| zx_status_t LoadSockAddr(const struct sockaddr* addr, size_t addr_len) { |
| // Address length larger than sockaddr_storage causes an error for API compatibility only. |
| if (addr == nullptr || addr_len > sizeof(struct sockaddr_storage)) { |
| return ZX_ERR_INVALID_ARGS; |
| } |
| switch (addr->sa_family) { |
| case AF_INET: { |
| if (addr_len < sizeof(struct sockaddr_in)) { |
| return ZX_ERR_INVALID_ARGS; |
| } |
| const auto& s = *reinterpret_cast<const struct sockaddr_in*>(addr); |
| fnet::wire::Ipv4SocketAddress address = { |
| .port = ntohs(s.sin_port), |
| }; |
| static_assert(sizeof(address.address.addr) == sizeof(s.sin_addr.s_addr), |
| "size of IPv4 addresses should be the same"); |
| memcpy(address.address.addr.data(), &s.sin_addr.s_addr, sizeof(s.sin_addr.s_addr)); |
| storage_ = address; |
| return ZX_OK; |
| } |
| case AF_INET6: { |
| if (addr_len < sizeof(struct sockaddr_in6)) { |
| return ZX_ERR_INVALID_ARGS; |
| } |
| const auto& s = *reinterpret_cast<const struct sockaddr_in6*>(addr); |
| fnet::wire::Ipv6SocketAddress address = { |
| .port = ntohs(s.sin6_port), |
| .zone_index = s.sin6_scope_id, |
| }; |
| static_assert( |
| decltype(address.address.addr)::size() == std::size(decltype(s.sin6_addr.s6_addr){}), |
| "size of IPv6 addresses should be the same"); |
| std::copy(std::begin(s.sin6_addr.s6_addr), std::end(s.sin6_addr.s6_addr), |
| address.address.addr.begin()); |
| storage_ = address; |
| return ZX_OK; |
| } |
| default: |
| return ZX_ERR_INVALID_ARGS; |
| } |
| } |
| |
| // Helpers from the reference documentation for std::visit<>, to allow |
| // visit-by-overload of the std::variant<> below: |
| template <class... Ts> |
| struct overloaded : Ts... { |
| using Ts::operator()...; |
| }; |
| // explicit deduction guide (not needed as of C++20) |
| template <class... Ts> |
| overloaded(Ts...) -> overloaded<Ts...>; |
| |
| template <typename F> |
| std::invoke_result_t<F, fnet::wire::SocketAddress> WithFIDL(F fn) { |
| return fn([this]() -> fnet::wire::SocketAddress { |
| if (storage_.has_value()) { |
| return std::visit( |
| overloaded{ |
| [](fnet::wire::Ipv4SocketAddress& ipv4) { |
| return fnet::wire::SocketAddress::WithIpv4( |
| fidl::ObjectView<fnet::wire::Ipv4SocketAddress>::FromExternal(&ipv4)); |
| }, |
| [](fnet::wire::Ipv6SocketAddress& ipv6) { |
| return fnet::wire::SocketAddress::WithIpv6( |
| fidl::ObjectView<fnet::wire::Ipv6SocketAddress>::FromExternal(&ipv6)); |
| }, |
| }, |
| storage_.value()); |
| } |
| return {}; |
| }()); |
| } |
| |
| private: |
| std::optional<std::variant<fnet::wire::Ipv4SocketAddress, fnet::wire::Ipv6SocketAddress>> |
| storage_; |
| }; |
| |
| // A helper structure to keep a packet info and any members' variants |
| // allocations on the stack. |
| struct PacketInfo { |
| zx_status_t LoadSockAddr(const sockaddr* addr, size_t addr_len) { |
| // Address length larger than sockaddr_storage causes an error for API compatibility only. |
| if (addr == nullptr || addr_len > sizeof(sockaddr_storage)) { |
| return ZX_ERR_INVALID_ARGS; |
| } |
| switch (addr->sa_family) { |
| case AF_PACKET: { |
| if (addr_len < sizeof(sockaddr_ll)) { |
| return ZX_ERR_INVALID_ARGS; |
| } |
| const auto& s = *reinterpret_cast<const sockaddr_ll*>(addr); |
| protocol_ = ntohs(s.sll_protocol); |
| interface_id_ = s.sll_ifindex; |
| switch (s.sll_halen) { |
| case 0: |
| eui48_storage_.reset(); |
| return ZX_OK; |
| case ETH_ALEN: { |
| fnet::wire::MacAddress address; |
| static_assert(decltype(address.octets)::size() == ETH_ALEN, |
| "eui48 address must have the same size as ETH_ALEN"); |
| static_assert(sizeof(s.sll_addr) == ETH_ALEN + 2); |
| memcpy(address.octets.data(), s.sll_addr, ETH_ALEN); |
| eui48_storage_ = address; |
| return ZX_OK; |
| } |
| default: |
| return ZX_ERR_NOT_SUPPORTED; |
| } |
| } |
| default: |
| return ZX_ERR_INVALID_ARGS; |
| } |
| } |
| |
| template <typename F> |
| std::invoke_result_t<F, fidl::ObjectView<fpacketsocket::wire::PacketInfo>> WithFIDL(F fn) { |
| auto packet_info = [this]() -> fpacketsocket::wire::PacketInfo { |
| return { |
| .protocol = protocol_, |
| .interface_id = interface_id_, |
| .addr = |
| [this]() { |
| if (eui48_storage_.has_value()) { |
| return fpacketsocket::wire::HardwareAddress::WithEui48( |
| fidl::ObjectView<fnet::wire::MacAddress>::FromExternal( |
| &eui48_storage_.value())); |
| } |
| return fpacketsocket::wire::HardwareAddress::WithNone({}); |
| }(), |
| }; |
| }(); |
| return fn(fidl::ObjectView<fpacketsocket::wire::PacketInfo>::FromExternal(&packet_info)); |
| } |
| |
| private: |
| decltype(fpacketsocket::wire::PacketInfo::protocol) protocol_; |
| decltype(fpacketsocket::wire::PacketInfo::interface_id) interface_id_; |
| std::optional<fnet::wire::MacAddress> eui48_storage_; |
| }; |
| |
| int16_t ParseSocketLevelControlMessage(fsocket::wire::SocketSendControlData& fidl_socket, int type, |
| const void* data, socklen_t len) { |
| // TODO(https://fxbug.dev/88984): Validate unsupported SOL_SOCKET control messages. |
| return 0; |
| } |
| |
| int16_t ParseIpLevelControlMessage(fsocket::wire::IpSendControlData& fidl_ip, int type, |
| const void* data, socklen_t len) { |
| switch (type) { |
| case IP_TTL: { |
| int ttl; |
| if (len != sizeof(ttl)) { |
| return EINVAL; |
| } |
| memcpy(&ttl, data, sizeof(ttl)); |
| if (ttl < 0 || ttl > std::numeric_limits<uint8_t>::max()) { |
| return EINVAL; |
| } |
| fidl_ip.set_ttl(static_cast<uint8_t>(ttl)); |
| return 0; |
| } |
| default: |
| // TODO(https://fxbug.dev/88984): Validate unsupported SOL_IP control messages. |
| return 0; |
| } |
| } |
| |
| int16_t ParseIpv6LevelControlMessage(fsocket::wire::Ipv6SendControlData& fidl_ipv6, |
| fidl::AnyArena& allocator, int type, const void* data, |
| socklen_t data_len) { |
| switch (type) { |
| case IPV6_HOPLIMIT: { |
| int hoplimit; |
| if (data_len != sizeof(hoplimit)) { |
| return EINVAL; |
| } |
| memcpy(&hoplimit, data, sizeof(hoplimit)); |
| if (hoplimit < -1 || hoplimit > std::numeric_limits<uint8_t>::max()) { |
| return EINVAL; |
| } |
| // Ignore hoplimit if it's -1 as it it is interpreted as if the cmsg was not present. |
| // |
| // https://github.com/torvalds/linux/blob/eaa54b1458c/net/ipv6/udp.c#L1531 |
| if (hoplimit != -1) { |
| fidl_ipv6.set_hoplimit(static_cast<uint8_t>(hoplimit)); |
| } |
| return 0; |
| } |
| case IPV6_PKTINFO: { |
| in6_pktinfo pktinfo; |
| if (data_len != sizeof(pktinfo)) { |
| return EINVAL; |
| } |
| memcpy(&pktinfo, data, sizeof(pktinfo)); |
| fsocket::wire::Ipv6PktInfoSendControlData fidl_pktinfo{ |
| .iface = static_cast<uint64_t>(pktinfo.ipi6_ifindex), |
| }; |
| static_assert(sizeof(pktinfo.ipi6_addr) == sizeof(fidl_pktinfo.local_addr.addr), |
| "mismatch between size of FIDL and in6_pktinfo IPv6 addresses"); |
| memcpy(fidl_pktinfo.local_addr.addr.data(), &pktinfo.ipi6_addr, sizeof(pktinfo.ipi6_addr)); |
| fidl_ipv6.set_pktinfo(allocator, fidl_pktinfo); |
| return 0; |
| } |
| default: |
| // TODO(https://fxbug.dev/88984): Validate unsupported SOL_IPV6 control messages. |
| return 0; |
| } |
| } |
| |
| int16_t ParseControlMessage(fsocket::wire::SocketSendControlData& fidl_socket, |
| fidl::AnyArena& allocator, int level, int type, const void* data, |
| socklen_t data_len) { |
| switch (level) { |
| case SOL_SOCKET: |
| return ParseSocketLevelControlMessage(fidl_socket, type, data, data_len); |
| default: |
| return 0; |
| } |
| } |
| |
| int16_t ParseControlMessage(fsocket::wire::NetworkSocketSendControlData& fidl_net, |
| fidl::AnyArena& allocator, int level, int type, const void* data, |
| socklen_t data_len) { |
| switch (level) { |
| case SOL_SOCKET: |
| if (!fidl_net.has_socket()) { |
| fidl_net.set_socket(allocator, fsocket::wire::SocketSendControlData(allocator)); |
| } |
| return ParseSocketLevelControlMessage(fidl_net.socket(), type, data, data_len); |
| case SOL_IP: |
| if (!fidl_net.has_ip()) { |
| fidl_net.set_ip(allocator, fsocket::wire::IpSendControlData(allocator)); |
| } |
| return ParseIpLevelControlMessage(fidl_net.ip(), type, data, data_len); |
| case SOL_IPV6: |
| if (!fidl_net.has_ipv6()) { |
| fidl_net.set_ipv6(allocator, fsocket::wire::Ipv6SendControlData(allocator)); |
| } |
| return ParseIpv6LevelControlMessage(fidl_net.ipv6(), allocator, type, data, data_len); |
| default: |
| return 0; |
| } |
| } |
| |
| int16_t ParseControlMessage(fsocket::wire::DatagramSocketSendControlData& fidl_dgram, |
| fidl::AnyArena& allocator, int level, int type, const void* data, |
| socklen_t data_len) { |
| if (!fidl_dgram.has_network()) { |
| fidl_dgram.set_network(allocator, fsocket::wire::NetworkSocketSendControlData(allocator)); |
| } |
| return ParseControlMessage(fidl_dgram.network(), allocator, level, type, data, data_len); |
| } |
| |
| int16_t ParseControlMessage(fpacketsocket::wire::SendControlData& fidl_packet, |
| fidl::AnyArena& allocator, int level, int type, const void* data, |
| socklen_t data_len) { |
| if (!fidl_packet.has_socket()) { |
| fidl_packet.set_socket(allocator, fsocket::wire::SocketSendControlData(allocator)); |
| } |
| return ParseControlMessage(fidl_packet.socket(), allocator, level, type, data, data_len); |
| } |
| |
| template <typename T> |
| fitx::result<int16_t, T> ParseControlMessages(const void* buf, socklen_t len, |
| fidl::AnyArena& allocator) { |
| if (buf == nullptr && len != 0) { |
| return fitx::error(static_cast<int16_t>(EFAULT)); |
| } |
| |
| T fidl_cmsg(allocator); |
| cpp20::span posix_cmsg(static_cast<const unsigned char*>(buf), len); |
| // Stop parsing once there is not enough bytes left to form a full cmsghdr. |
| // https://github.com/torvalds/linux/blob/42eb8fdac2f/net/core/sock.c#L2644 |
| // https://github.com/torvalds/linux/blob/42eb8fdac2f/include/linux/socket.h#L115-L126 |
| while (posix_cmsg.size() >= sizeof(cmsghdr)) { |
| // Do not access the control buffer directly, as it may be misaligned. |
| cmsghdr cmsg; |
| memcpy(&cmsg, posix_cmsg.data(), sizeof(cmsg)); |
| |
| // Validate the header length. |
| // https://github.com/torvalds/linux/blob/42eb8fdac2f/include/linux/socket.h#L119-L122 |
| if (cmsg.cmsg_len < sizeof(cmsg) || cmsg.cmsg_len > posix_cmsg.size()) { |
| return fitx::error(static_cast<int16_t>(EINVAL)); |
| } |
| const void* data = CMSG_DATA(posix_cmsg.data()); |
| const socklen_t data_len = cmsg.cmsg_len - CMSG_ALIGN(sizeof(cmsghdr)); |
| ZX_ASSERT_MSG(reinterpret_cast<const unsigned char*>(data) + data_len < posix_cmsg.end(), |
| "incoherent data buffer bounds, %p + %x > %p", data, data_len, posix_cmsg.end()); |
| posix_cmsg = posix_cmsg.subspan(cmsg.cmsg_len); |
| |
| int16_t err = |
| ParseControlMessage(fidl_cmsg, allocator, cmsg.cmsg_level, cmsg.cmsg_type, data, data_len); |
| if (err != 0) { |
| return fitx::error(err); |
| } |
| } |
| |
| return fitx::success(fidl_cmsg); |
| } |
| |
| class FidlControlDataProcessor { |
| public: |
| FidlControlDataProcessor(void* buf, socklen_t len) |
| : buffer_(cpp20::span{reinterpret_cast<unsigned char*>(buf), len}) {} |
| |
| socklen_t Store(fsocket::wire::DatagramSocketRecvControlData const& control_data) { |
| socklen_t total = 0; |
| if (control_data.has_network()) { |
| total += Store(control_data.network()); |
| } |
| return total; |
| } |
| |
| socklen_t Store(fsocket::wire::NetworkSocketRecvControlData const& control_data) { |
| socklen_t total = 0; |
| if (control_data.has_socket()) { |
| total += Store(control_data.socket()); |
| } |
| if (control_data.has_ip()) { |
| total += Store(control_data.ip()); |
| } |
| if (control_data.has_ipv6()) { |
| total += Store(control_data.ipv6()); |
| } |
| return total; |
| } |
| |
| socklen_t Store(fpacketsocket::wire::RecvControlData const& control_data) { |
| socklen_t total = 0; |
| if (control_data.has_socket()) { |
| total += Store(control_data.socket()); |
| } |
| return total; |
| } |
| |
| private: |
| socklen_t Store(fsocket::wire::SocketRecvControlData const& control_data) { |
| socklen_t total = 0; |
| |
| if (control_data.has_timestamp()) { |
| const fsocket::wire::Timestamp& timestamp = control_data.timestamp(); |
| |
| std::chrono::duration t = std::chrono::nanoseconds(timestamp.nanoseconds); |
| std::chrono::seconds sec = std::chrono::duration_cast<std::chrono::seconds>(t); |
| switch (timestamp.requested) { |
| case fsocket::wire::TimestampOption::kNanosecond: { |
| const struct timespec ts = { |
| .tv_sec = sec.count(), |
| .tv_nsec = std::chrono::duration_cast<std::chrono::nanoseconds>(t - sec).count(), |
| }; |
| total += StoreControlMessage(SOL_SOCKET, SO_TIMESTAMPNS, &ts, sizeof(ts)); |
| } break; |
| case fsocket::wire::TimestampOption::kMicrosecond: { |
| const struct timeval tv = { |
| .tv_sec = sec.count(), |
| .tv_usec = std::chrono::duration_cast<std::chrono::microseconds>(t - sec).count(), |
| }; |
| total += StoreControlMessage(SOL_SOCKET, SO_TIMESTAMP, &tv, sizeof(tv)); |
| } break; |
| case fsocket::wire::TimestampOption::kDisabled: |
| break; |
| } |
| } |
| |
| return total; |
| } |
| |
| socklen_t Store(fsocket::wire::IpRecvControlData const& control_data) { |
| socklen_t total = 0; |
| if (control_data.has_tos()) { |
| const uint8_t tos = control_data.tos(); |
| total += StoreControlMessage(IPPROTO_IP, IP_TOS, &tos, sizeof(tos)); |
| } |
| if (control_data.has_ttl()) { |
| // Even though the ttl can be encoded in a single byte, Linux returns it as an `int` when it |
| // is received as a control message. |
| // https://github.com/torvalds/linux/blob/7e57714cd0a/net/ipv4/ip_sockglue.c#L67 |
| const int ttl = static_cast<int>(control_data.ttl()); |
| total += StoreControlMessage(IPPROTO_IP, IP_TTL, &ttl, sizeof(ttl)); |
| } |
| return total; |
| } |
| |
| socklen_t Store(fsocket::wire::Ipv6RecvControlData const& control_data) { |
| socklen_t total = 0; |
| if (control_data.has_tclass()) { |
| // Even though the traffic class can be encoded in a single byte, Linux returns it as an `int` |
| // when it is received as a control message. |
| // https://github.com/torvalds/linux/blob/7e57714cd0a/include/net/ipv6.h#L968 |
| const int tclass = static_cast<int>(control_data.tclass()); |
| total += StoreControlMessage(IPPROTO_IPV6, IPV6_TCLASS, &tclass, sizeof(tclass)); |
| } |
| if (control_data.has_hoplimit()) { |
| // Even though the hop limit can be encoded in a single byte, Linux returns it as an `int` |
| // when it is received as a control message. |
| // https://github.com/torvalds/linux/blob/7e57714cd0a/net/ipv6/datagram.c#L622 |
| const int hoplimit = static_cast<int>(control_data.hoplimit()); |
| total += StoreControlMessage(IPPROTO_IPV6, IPV6_HOPLIMIT, &hoplimit, sizeof(hoplimit)); |
| } |
| if (control_data.has_pktinfo()) { |
| const fsocket::wire::Ipv6PktInfoRecvControlData& fidl_pktinfo = control_data.pktinfo(); |
| in6_pktinfo pktinfo = { |
| .ipi6_ifindex = static_cast<unsigned int>(fidl_pktinfo.iface), |
| }; |
| static_assert( |
| sizeof(pktinfo.ipi6_addr) == decltype(fidl_pktinfo.header_destination_addr.addr)::size(), |
| "mismatch between size of FIDL and in6_pktinfo IPv6 addresses"); |
| memcpy(&pktinfo.ipi6_addr, fidl_pktinfo.header_destination_addr.addr.data(), |
| sizeof(pktinfo.ipi6_addr)); |
| total += StoreControlMessage(IPPROTO_IPV6, IPV6_PKTINFO, &pktinfo, sizeof(pktinfo)); |
| } |
| return total; |
| } |
| |
| socklen_t StoreControlMessage(int level, int type, const void* data, socklen_t len) { |
| socklen_t cmsg_len = CMSG_LEN(len); |
| size_t bytes_left = buffer_.size(); |
| if (bytes_left < cmsg_len) { |
| // Not enough space to store the entire control message. |
| // TODO(https://fxbug.dev/86146): Add support for truncated control messages (MSG_CTRUNC). |
| return 0; |
| } |
| |
| // The user-provided pointer is not guaranteed to be aligned. So instead of casting it into a |
| // struct cmsghdr and writing to it directly, stack-allocate one and then copy it. |
| cmsghdr cmsg = { |
| .cmsg_len = cmsg_len, |
| .cmsg_level = level, |
| .cmsg_type = type, |
| }; |
| unsigned char* buf = buffer_.data(); |
| ZX_ASSERT_MSG(CMSG_DATA(buf) + len <= buf + bytes_left, |
| "buffer would overflow, %p + %x > %p + %zx", CMSG_DATA(buf), len, buf, |
| bytes_left); |
| memcpy(buf, &cmsg, sizeof(cmsg)); |
| memcpy(CMSG_DATA(buf), data, len); |
| size_t bytes_consumed = std::min(CMSG_SPACE(len), bytes_left); |
| buffer_ = buffer_.subspan(bytes_consumed); |
| |
| return static_cast<socklen_t>(bytes_consumed); |
| } |
| |
| cpp20::span<unsigned char> buffer_; |
| }; |
| |
| fsocket::wire::RecvMsgFlags to_recvmsg_flags(int flags) { |
| fsocket::wire::RecvMsgFlags r; |
| if (flags & MSG_PEEK) { |
| r |= fsocket::wire::RecvMsgFlags::kPeek; |
| } |
| return r; |
| } |
| |
| fsocket::wire::SendMsgFlags to_sendmsg_flags(int flags) { return fsocket::wire::SendMsgFlags(); } |
| |
| socklen_t fidl_to_sockaddr(const fnet::wire::SocketAddress& fidl, void* addr, socklen_t addr_len) { |
| switch (fidl.Which()) { |
| case fnet::wire::SocketAddress::Tag::kIpv4: { |
| const auto& ipv4 = fidl.ipv4(); |
| struct sockaddr_in tmp = { |
| .sin_family = AF_INET, |
| .sin_port = htons(ipv4.port), |
| }; |
| static_assert(sizeof(tmp.sin_addr.s_addr) == sizeof(ipv4.address.addr), |
| "size of IPv4 addresses should be the same"); |
| memcpy(&tmp.sin_addr.s_addr, ipv4.address.addr.data(), sizeof(ipv4.address.addr)); |
| // Copy truncated address. |
| memcpy(addr, &tmp, std::min(sizeof(tmp), static_cast<size_t>(addr_len))); |
| return sizeof(tmp); |
| } |
| case fnet::wire::SocketAddress::Tag::kIpv6: { |
| const auto& ipv6 = fidl.ipv6(); |
| struct sockaddr_in6 tmp = { |
| .sin6_family = AF_INET6, |
| .sin6_port = htons(ipv6.port), |
| .sin6_scope_id = static_cast<uint32_t>(ipv6.zone_index), |
| }; |
| static_assert(std::size(tmp.sin6_addr.s6_addr) == decltype(ipv6.address.addr)::size(), |
| "size of IPv6 addresses should be the same"); |
| std::copy(ipv6.address.addr.begin(), ipv6.address.addr.end(), |
| std::begin(tmp.sin6_addr.s6_addr)); |
| // Copy truncated address. |
| memcpy(addr, &tmp, std::min(sizeof(tmp), static_cast<size_t>(addr_len))); |
| return sizeof(tmp); |
| } |
| } |
| } |
| |
| uint16_t fidl_protoassoc_to_protocol(const fpacketsocket::wire::ProtocolAssociation& protocol) { |
| // protocol has an invalid tag when it's not provided by the server (when the socket is not |
| // associated). |
| // |
| // TODO(https://fxbug.dev/58503): Use better representation of nullable union when available. |
| if (protocol.has_invalid_tag()) { |
| return 0; |
| } |
| |
| switch (protocol.Which()) { |
| case fpacketsocket::wire::ProtocolAssociation::Tag::kAll: |
| return ETH_P_ALL; |
| case fpacketsocket::wire::ProtocolAssociation::Tag::kSpecified: |
| return protocol.specified(); |
| } |
| } |
| |
| void populate_from_fidl_hwaddr(const fpacketsocket::wire::HardwareAddress& addr, sockaddr_ll& s) { |
| switch (addr.Which()) { |
| case fpacketsocket::wire::HardwareAddress::Tag::kUnknown: |
| // The server is newer than us and sending a variant we don't understand. |
| __FALLTHROUGH; |
| case fpacketsocket::wire::HardwareAddress::Tag::kNone: |
| s.sll_halen = 0; |
| break; |
| case fpacketsocket::wire::HardwareAddress::Tag::kEui48: { |
| const fnet::wire::MacAddress& eui48 = addr.eui48(); |
| static_assert(std::size(decltype(s.sll_addr){}) == decltype(eui48.octets)::size() + 2); |
| std::copy(eui48.octets.begin(), eui48.octets.end(), std::begin(s.sll_addr)); |
| s.sll_halen = decltype(eui48.octets)::size(); |
| } break; |
| } |
| } |
| |
| uint16_t fidl_hwtype_to_arphrd(const fpacketsocket::wire::HardwareType type) { |
| switch (type) { |
| case fpacketsocket::wire::HardwareType::kNetworkOnly: |
| return ARPHRD_NONE; |
| case fpacketsocket::wire::HardwareType::kEthernet: |
| return ARPHRD_ETHER; |
| case fpacketsocket::wire::HardwareType::kLoopback: |
| return ARPHRD_LOOPBACK; |
| } |
| } |
| |
| uint8_t fidl_pkttype_to_pkttype(const fpacketsocket::wire::PacketType type) { |
| switch (type) { |
| case fpacketsocket::wire::PacketType::kHost: |
| return PACKET_HOST; |
| case fpacketsocket::wire::PacketType::kBroadcast: |
| return PACKET_BROADCAST; |
| case fpacketsocket::wire::PacketType::kMulticast: |
| return PACKET_MULTICAST; |
| case fpacketsocket::wire::PacketType::kOtherHost: |
| return PACKET_OTHERHOST; |
| case fpacketsocket::wire::PacketType::kOutgoing: |
| return PACKET_OUTGOING; |
| } |
| } |
| |
| // https://github.com/torvalds/linux/blob/f2850dd5ee0/include/net/tcp.h#L1012 |
| constexpr socklen_t kTcpCANameMax = 16; |
| constexpr const char kCcCubic[kTcpCANameMax] = "cubic"; |
| constexpr const char kCcReno[kTcpCANameMax] = "reno"; |
| |
| struct SockOptResult { |
| const zx_status_t status; |
| const int16_t err; |
| |
| bool ok() const { return status == ZX_OK && err == 0; } |
| |
| static inline SockOptResult Ok() { return SockOptResult{ZX_OK, 0}; } |
| |
| static inline SockOptResult Errno(int16_t err) { return SockOptResult{ZX_OK, err}; } |
| |
| static inline SockOptResult Zx(zx_status_t status) { return SockOptResult{status, 0}; } |
| |
| template <typename T> |
| static inline SockOptResult FromFidlResponse(const T& response) { |
| if (response.status() != ZX_OK) { |
| return SockOptResult::Zx(response.status()); |
| } |
| const auto& result = response.value(); |
| if (result.is_error()) { |
| return SockOptResult::Errno(static_cast<int16_t>(result.error_value())); |
| } |
| return SockOptResult::Ok(); |
| } |
| }; |
| |
| class GetSockOptProcessor { |
| public: |
| GetSockOptProcessor(void* optval, socklen_t* optlen) : optval_(optval), optlen_(optlen) {} |
| |
| template <typename T, typename F> |
| SockOptResult Process(T&& response, F getter) { |
| if (response.status() != ZX_OK) { |
| return SockOptResult::Zx(response.status()); |
| } |
| const auto& result = response.value(); |
| if (result.is_error()) { |
| return SockOptResult::Errno(static_cast<int16_t>(result.error_value())); |
| } |
| return StoreOption(getter(*result.value())); |
| } |
| |
| template <typename T> |
| SockOptResult StoreOption(const T& value) { |
| static_assert(sizeof(T) != sizeof(T), "function must be specialized"); |
| } |
| |
| private: |
| SockOptResult StoreRaw(const void* data, socklen_t data_len) { |
| if (data_len > *optlen_) { |
| return SockOptResult::Errno(EINVAL); |
| } |
| memcpy(optval_, data, data_len); |
| *optlen_ = data_len; |
| return SockOptResult::Ok(); |
| } |
| |
| void* const optval_; |
| socklen_t* const optlen_; |
| }; |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const int32_t& value) { |
| return StoreRaw(&value, sizeof(int32_t)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const uint32_t& value) { |
| return StoreRaw(&value, sizeof(uint32_t)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const uint8_t& value) { |
| return StoreRaw(&value, sizeof(uint8_t)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fsocket::wire::Domain& value) { |
| int32_t domain; |
| switch (value) { |
| case fsocket::wire::Domain::kIpv4: |
| domain = AF_INET; |
| break; |
| case fsocket::wire::Domain::kIpv6: |
| domain = AF_INET6; |
| break; |
| } |
| return StoreOption(domain); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const bool& value) { |
| return StoreOption(static_cast<uint32_t>(value)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const struct linger& value) { |
| return StoreRaw(&value, sizeof(struct linger)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fidl::StringView& value) { |
| if (value.empty()) { |
| *optlen_ = 0; |
| } else if (*optlen_ > value.size()) { |
| char* p = std::copy(value.begin(), value.end(), static_cast<char*>(optval_)); |
| *p = 0; |
| *optlen_ = static_cast<socklen_t>(value.size()) + 1; |
| } else { |
| return SockOptResult::Errno(EINVAL); |
| } |
| return SockOptResult::Ok(); |
| } |
| |
| // Helper type to provide GetSockOptProcessor with a truncating string view conversion. |
| struct TruncatingStringView { |
| explicit TruncatingStringView(fidl::StringView string) : string(string) {} |
| |
| fidl::StringView string; |
| }; |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const TruncatingStringView& value) { |
| *optlen_ = std::min(*optlen_, static_cast<socklen_t>(value.string.size())); |
| char* p = std::copy_n(value.string.begin(), *optlen_ - 1, static_cast<char*>(optval_)); |
| *p = 0; |
| return SockOptResult::Ok(); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fsocket::wire::OptionalUint8& value) { |
| switch (value.Which()) { |
| case fsocket::wire::OptionalUint8::Tag::kValue: |
| return StoreOption(static_cast<int32_t>(value.value())); |
| case fsocket::wire::OptionalUint8::Tag::kUnset: |
| return StoreOption(-1); |
| } |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fsocket::wire::OptionalUint32& value) { |
| switch (value.Which()) { |
| case fsocket::wire::OptionalUint32::Tag::kValue: |
| ZX_ASSERT(value.value() < std::numeric_limits<int32_t>::max()); |
| return StoreOption(static_cast<int32_t>(value.value())); |
| case fsocket::wire::OptionalUint32::Tag::kUnset: |
| return StoreOption(-1); |
| } |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fnet::wire::Ipv4Address& value) { |
| static_assert(sizeof(struct in_addr) == sizeof(value.addr)); |
| return StoreRaw(value.addr.data(), sizeof(value.addr)); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const frawsocket::wire::Icmpv6Filter& value) { |
| static_assert(sizeof(icmp6_filter) == sizeof(value.blocked_types)); |
| *optlen_ = std::min(static_cast<socklen_t>(sizeof(icmp6_filter)), *optlen_); |
| memcpy(optval_, value.blocked_types.data(), *optlen_); |
| return SockOptResult::Ok(); |
| } |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const fsocket::wire::TcpInfo& value) { |
| tcp_info info; |
| // Explicitly initialize unsupported fields to a garbage value. It would probably be quieter to |
| // zero-initialize, but that can mask bugs in the interpretation of fields for which zero is a |
| // valid value. |
| // |
| // Note that "unsupported" includes fields not defined in FIDL *and* fields not populated by the |
| // server. |
| memset(&info, 0xff, sizeof(info)); |
| |
| if (value.has_state()) { |
| info.tcpi_state = [](fsocket::wire::TcpState state) -> uint8_t { |
| switch (state) { |
| case fsocket::wire::TcpState::kEstablished: |
| return TCP_ESTABLISHED; |
| case fsocket::wire::TcpState::kSynSent: |
| return TCP_SYN_SENT; |
| case fsocket::wire::TcpState::kSynRecv: |
| return TCP_SYN_RECV; |
| case fsocket::wire::TcpState::kFinWait1: |
| return TCP_FIN_WAIT1; |
| case fsocket::wire::TcpState::kFinWait2: |
| return TCP_FIN_WAIT2; |
| case fsocket::wire::TcpState::kTimeWait: |
| return TCP_TIME_WAIT; |
| case fsocket::wire::TcpState::kClose: |
| return TCP_CLOSE; |
| case fsocket::wire::TcpState::kCloseWait: |
| return TCP_CLOSE_WAIT; |
| case fsocket::wire::TcpState::kLastAck: |
| return TCP_LAST_ACK; |
| case fsocket::wire::TcpState::kListen: |
| return TCP_LISTEN; |
| case fsocket::wire::TcpState::kClosing: |
| return TCP_CLOSING; |
| } |
| }(value.state()); |
| } |
| if (value.has_ca_state()) { |
| info.tcpi_ca_state = [](fsocket::wire::TcpCongestionControlState ca_state) -> uint8_t { |
| switch (ca_state) { |
| case fsocket::wire::TcpCongestionControlState::kOpen: |
| return TCP_CA_Open; |
| case fsocket::wire::TcpCongestionControlState::kDisorder: |
| return TCP_CA_Disorder; |
| case fsocket::wire::TcpCongestionControlState::kCongestionWindowReduced: |
| return TCP_CA_CWR; |
| case fsocket::wire::TcpCongestionControlState::kRecovery: |
| return TCP_CA_Recovery; |
| case fsocket::wire::TcpCongestionControlState::kLoss: |
| return TCP_CA_Loss; |
| } |
| }(value.ca_state()); |
| } |
| if (value.has_rto_usec()) { |
| info.tcpi_rto = value.rto_usec(); |
| } |
| if (value.has_rtt_usec()) { |
| info.tcpi_rtt = value.rtt_usec(); |
| } |
| if (value.has_rtt_var_usec()) { |
| info.tcpi_rttvar = value.rtt_var_usec(); |
| } |
| if (value.has_snd_ssthresh()) { |
| info.tcpi_snd_ssthresh = value.snd_ssthresh(); |
| } |
| if (value.has_snd_cwnd()) { |
| info.tcpi_snd_cwnd = value.snd_cwnd(); |
| } |
| if (value.has_reorder_seen()) { |
| info.tcpi_reord_seen = value.reorder_seen(); |
| } |
| |
| static_assert(sizeof(info) <= std::numeric_limits<socklen_t>::max()); |
| return StoreRaw(&info, std::min(*optlen_, static_cast<socklen_t>(sizeof(info)))); |
| } |
| |
| // Used for various options that allow the caller to supply larger buffers than needed. |
| struct PartialCopy { |
| int32_t value; |
| // Appears to be true for IP_*, SO_* and false for IPV6_*. |
| bool allow_char; |
| }; |
| |
| template <> |
| SockOptResult GetSockOptProcessor::StoreOption(const PartialCopy& value) { |
| socklen_t want_size = |
| *optlen_ < sizeof(int32_t) && value.allow_char ? sizeof(uint8_t) : sizeof(value.value); |
| *optlen_ = std::min(want_size, *optlen_); |
| memcpy(optval_, &value.value, *optlen_); |
| return SockOptResult::Ok(); |
| } |
| |
| class SetSockOptProcessor { |
| public: |
| SetSockOptProcessor(const void* optval, socklen_t optlen) : optval_(optval), optlen_(optlen) {} |
| |
| template <typename T> |
| int16_t Get(T& out) { |
| if (optlen_ < sizeof(T)) { |
| return EINVAL; |
| } |
| memcpy(&out, optval_, sizeof(T)); |
| return 0; |
| } |
| |
| template <typename T, typename F> |
| SockOptResult Process(F f) { |
| T v; |
| int16_t result = Get(v); |
| if (result) { |
| return SockOptResult::Errno(result); |
| } |
| return SockOptResult::FromFidlResponse(f(std::move(v))); |
| } |
| |
| private: |
| const void* const optval_; |
| socklen_t const optlen_; |
| }; |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(fidl::StringView& out) { |
| const char* optval = static_cast<const char*>(optval_); |
| out = fidl::StringView::FromExternal(optval, strnlen(optval, optlen_)); |
| return 0; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(bool& out) { |
| int32_t i; |
| int16_t r = Get(i); |
| out = i != 0; |
| return r; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(uint32_t& out) { |
| int32_t& alt = *reinterpret_cast<int32_t*>(&out); |
| if (int16_t r = Get(alt); r) { |
| return r; |
| } |
| if (alt < 0) { |
| return EINVAL; |
| } |
| return 0; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(fsocket::wire::OptionalUint8& out) { |
| int32_t i; |
| if (int16_t r = Get(i); r) { |
| return r; |
| } |
| if (i < -1 || i > std::numeric_limits<uint8_t>::max()) { |
| return EINVAL; |
| } |
| if (i == -1) { |
| out = fsocket::wire::OptionalUint8::WithUnset({}); |
| } else { |
| out = fsocket::wire::OptionalUint8::WithValue(static_cast<uint8_t>(i)); |
| } |
| return 0; |
| } |
| |
| // Like OptionalUint8, but permits truncation to a single byte. |
| struct OptionalUint8CharAllowed { |
| fsocket::wire::OptionalUint8 inner; |
| }; |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(OptionalUint8CharAllowed& out) { |
| if (optlen_ == sizeof(uint8_t)) { |
| out.inner = fsocket::wire::OptionalUint8::WithValue(*static_cast<const uint8_t*>(optval_)); |
| return 0; |
| } |
| return Get(out.inner); |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(fsocket::wire::IpMulticastMembership& out) { |
| union { |
| struct ip_mreqn reqn; |
| struct ip_mreq req; |
| } r; |
| struct in_addr* local; |
| struct in_addr* mcast; |
| if (optlen_ < sizeof(struct ip_mreqn)) { |
| if (Get(r.req) != 0) { |
| return EINVAL; |
| } |
| out.iface = 0; |
| local = &r.req.imr_interface; |
| mcast = &r.req.imr_multiaddr; |
| } else { |
| if (Get(r.reqn) != 0) { |
| return EINVAL; |
| } |
| out.iface = r.reqn.imr_ifindex; |
| local = &r.reqn.imr_address; |
| mcast = &r.reqn.imr_multiaddr; |
| } |
| static_assert(sizeof(out.local_addr.addr) == sizeof(*local)); |
| memcpy(out.local_addr.addr.data(), local, sizeof(*local)); |
| static_assert(sizeof(out.mcast_addr.addr) == sizeof(*mcast)); |
| memcpy(out.mcast_addr.addr.data(), mcast, sizeof(*mcast)); |
| return 0; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(fsocket::wire::Ipv6MulticastMembership& out) { |
| struct ipv6_mreq req; |
| if (Get(req) != 0) { |
| return EINVAL; |
| } |
| out.iface = req.ipv6mr_interface; |
| static_assert(std::size(req.ipv6mr_multiaddr.s6_addr) == decltype(out.mcast_addr.addr)::size()); |
| std::copy(std::begin(req.ipv6mr_multiaddr.s6_addr), std::end(req.ipv6mr_multiaddr.s6_addr), |
| out.mcast_addr.addr.begin()); |
| return 0; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(frawsocket::wire::Icmpv6Filter& out) { |
| struct icmp6_filter filter; |
| if (Get(filter) != 0) { |
| return EINVAL; |
| } |
| |
| static_assert(sizeof(filter) == sizeof(out.blocked_types)); |
| memcpy(out.blocked_types.data(), &filter, sizeof(filter)); |
| return 0; |
| } |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(fsocket::wire::TcpCongestionControl& out) { |
| if (strncmp(static_cast<const char*>(optval_), kCcCubic, optlen_) == 0) { |
| out = fsocket::wire::TcpCongestionControl::kCubic; |
| return 0; |
| } |
| if (strncmp(static_cast<const char*>(optval_), kCcReno, optlen_) == 0) { |
| out = fsocket::wire::TcpCongestionControl::kReno; |
| return 0; |
| } |
| return ENOENT; |
| } |
| |
| struct IntOrChar { |
| int32_t value; |
| }; |
| |
| template <> |
| int16_t SetSockOptProcessor::Get(IntOrChar& out) { |
| if (Get(out.value) == 0) { |
| return 0; |
| } |
| if (optlen_ == 0) { |
| return EINVAL; |
| } |
| out.value = *static_cast<const uint8_t*>(optval_); |
| return 0; |
| } |
| |
| template <typename T, |
| typename = std::enable_if_t< |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fpacketsocket::Socket>>>> |
| struct BaseSocket { |
| static_assert(std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fpacketsocket::Socket>>); |
| |
| public: |
| explicit BaseSocket(T& client) : client_(client) {} |
| |
| T& client() { return client_; } |
| |
| SockOptResult get_solsocket_sockopt_fidl(int optname, void* optval, socklen_t* optlen) { |
| GetSockOptProcessor proc(optval, optlen); |
| switch (optname) { |
| case SO_TYPE: |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>>) { |
| return proc.StoreOption<int32_t>(SOCK_DGRAM); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>>) { |
| return proc.StoreOption<int32_t>(SOCK_STREAM); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>>) { |
| return proc.StoreOption<int32_t>(SOCK_RAW); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fpacketsocket::Socket>>) { |
| return proc.Process(client()->GetInfo(), [](const auto& response) { |
| switch (response.kind) { |
| case fpacketsocket::wire::Kind::kNetwork: |
| return SOCK_DGRAM; |
| case fpacketsocket::wire::Kind::kLink: |
| return SOCK_RAW; |
| } |
| }); |
| } |
| case SO_DOMAIN: |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fpacketsocket::Socket>>) { |
| return proc.StoreOption<int32_t>(AF_PACKET); |
| } else { |
| return proc.Process(client()->GetInfo(), |
| [](const auto& response) { return response.domain; }); |
| } |
| case SO_TIMESTAMP: |
| return proc.Process(client()->GetTimestamp(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value == fsocket::wire::TimestampOption::kMicrosecond, |
| .allow_char = false, |
| }; |
| }); |
| case SO_TIMESTAMPNS: |
| return proc.Process(client()->GetTimestamp(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value == fsocket::wire::TimestampOption::kNanosecond, |
| .allow_char = false, |
| }; |
| }); |
| case SO_PROTOCOL: |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>>) { |
| return proc.Process(client()->GetInfo(), [](const auto& response) { |
| switch (response.proto) { |
| case fsocket::wire::DatagramSocketProtocol::kUdp: |
| return IPPROTO_UDP; |
| case fsocket::wire::DatagramSocketProtocol::kIcmpEcho: |
| switch (response.domain) { |
| case fsocket::wire::Domain::kIpv4: |
| return IPPROTO_ICMP; |
| case fsocket::wire::Domain::kIpv6: |
| return IPPROTO_ICMPV6; |
| } |
| } |
| }); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>>) { |
| return proc.Process(client()->GetInfo(), [](const auto& response) { |
| switch (response.proto) { |
| case fsocket::wire::StreamSocketProtocol::kTcp: |
| return IPPROTO_TCP; |
| } |
| }); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>>) { |
| return proc.Process(client()->GetInfo(), [](const auto& response) { |
| switch (response.proto.Which()) { |
| case frawsocket::wire::ProtocolAssociation::Tag::kUnassociated: |
| return IPPROTO_RAW; |
| case frawsocket::wire::ProtocolAssociation::Tag::kAssociated: |
| return static_cast<int>(response.proto.associated()); |
| } |
| }); |
| } |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fpacketsocket::Socket>>) { |
| return proc.StoreOption<int32_t>(0); |
| } |
| case SO_ERROR: { |
| auto response = client()->GetError(); |
| if (response.status() != ZX_OK) { |
| return SockOptResult::Zx(response.status()); |
| } |
| int32_t error_code = 0; |
| const auto& result = response.value(); |
| if (result.is_error()) { |
| error_code = static_cast<int32_t>(result.error_value()); |
| } |
| return proc.StoreOption(error_code); |
| } |
| case SO_SNDBUF: |
| return proc.Process(client()->GetSendBuffer(), [](const auto& response) { |
| return static_cast<uint32_t>(response.value_bytes); |
| }); |
| case SO_RCVBUF: |
| return proc.Process(client()->GetReceiveBuffer(), [](const auto& response) { |
| return static_cast<uint32_t>(response.value_bytes); |
| }); |
| case SO_REUSEADDR: |
| return proc.Process(client()->GetReuseAddress(), |
| [](const auto& response) { return response.value; }); |
| case SO_REUSEPORT: |
| return proc.Process(client()->GetReusePort(), |
| [](const auto& response) { return response.value; }); |
| case SO_BINDTODEVICE: |
| return proc.Process( |
| client()->GetBindToDevice(), |
| [](auto& response) -> const fidl::StringView& { return response.value; }); |
| case SO_BROADCAST: |
| return proc.Process(client()->GetBroadcast(), |
| [](const auto& response) { return response.value; }); |
| case SO_KEEPALIVE: |
| return proc.Process(client()->GetKeepAlive(), |
| [](const auto& response) { return response.value; }); |
| case SO_LINGER: |
| return proc.Process(client()->GetLinger(), [](const auto& response) { |
| struct linger l; |
| l.l_onoff = response.linger; |
| // NB: l_linger is typed as int but interpreted as unsigned by |
| // linux. |
| l.l_linger = static_cast<int>(response.length_secs); |
| return l; |
| }); |
| case SO_ACCEPTCONN: |
| return proc.Process(client()->GetAcceptConn(), |
| [](const auto& response) { return response.value; }); |
| case SO_OOBINLINE: |
| return proc.Process(client()->GetOutOfBandInline(), |
| [](const auto& response) { return response.value; }); |
| case SO_NO_CHECK: |
| return proc.Process(client()->GetNoCheck(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case SO_SNDTIMEO: |
| case SO_RCVTIMEO: |
| case SO_PEERCRED: |
| return SockOptResult::Errno(EOPNOTSUPP); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| } |
| |
| SockOptResult set_solsocket_sockopt_fidl(int optname, const void* optval, socklen_t optlen) { |
| SetSockOptProcessor proc(optval, optlen); |
| switch (optname) { |
| case SO_TIMESTAMP: |
| return proc.Process<bool>([this](bool value) { |
| using fsocket::wire::TimestampOption; |
| TimestampOption opt = value ? TimestampOption::kMicrosecond : TimestampOption::kDisabled; |
| return client()->SetTimestamp(opt); |
| }); |
| case SO_TIMESTAMPNS: |
| return proc.Process<bool>([this](bool value) { |
| using fsocket::wire::TimestampOption; |
| TimestampOption opt = value ? TimestampOption::kNanosecond : TimestampOption::kDisabled; |
| return client()->SetTimestamp(opt); |
| }); |
| case SO_SNDBUF: |
| return proc.Process<int32_t>([this](int32_t value) { |
| // NB: SNDBUF treated as unsigned, we just cast the value to skip sign check. |
| return client()->SetSendBuffer(static_cast<uint64_t>(value)); |
| }); |
| case SO_RCVBUF: |
| return proc.Process<int32_t>([this](int32_t value) { |
| // NB: RCVBUF treated as unsigned, we just cast the value to skip sign check. |
| return client()->SetReceiveBuffer(static_cast<uint64_t>(value)); |
| }); |
| case SO_REUSEADDR: |
| return proc.Process<bool>([this](bool value) { return client()->SetReuseAddress(value); }); |
| case SO_REUSEPORT: |
| return proc.Process<bool>([this](bool value) { return client()->SetReusePort(value); }); |
| case SO_BINDTODEVICE: |
| return proc.Process<fidl::StringView>( |
| [this](fidl::StringView value) { return client()->SetBindToDevice(value); }); |
| case SO_BROADCAST: |
| return proc.Process<bool>([this](bool value) { return client()->SetBroadcast(value); }); |
| case SO_KEEPALIVE: |
| return proc.Process<bool>([this](bool value) { return client()->SetKeepAlive(value); }); |
| case SO_LINGER: |
| return proc.Process<struct linger>([this](struct linger value) { |
| // NB: l_linger is typed as int but interpreted as unsigned by linux. |
| return client()->SetLinger(value.l_onoff != 0, static_cast<uint32_t>(value.l_linger)); |
| }); |
| case SO_OOBINLINE: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetOutOfBandInline(value); }); |
| case SO_NO_CHECK: |
| return proc.Process<bool>([this](bool value) { return client()->SetNoCheck(value); }); |
| case SO_SNDTIMEO: |
| case SO_RCVTIMEO: |
| return SockOptResult::Errno(ENOTSUP); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| } |
| |
| private: |
| T& client_; |
| }; |
| |
| template <typename T, |
| typename = std::enable_if_t< |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>>>> |
| struct BaseNetworkSocket : public BaseSocket<T> { |
| static_assert(std::is_same_v<T, fidl::WireSyncClient<fsocket::SynchronousDatagramSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>> || |
| std::is_same_v<T, fidl::WireSyncClient<frawsocket::Socket>>); |
| |
| public: |
| using BaseSocket = BaseSocket<T>; |
| using BaseSocket::client; |
| |
| explicit BaseNetworkSocket(T& client) : BaseSocket(client) {} |
| |
| zx_status_t bind(const struct sockaddr* addr, socklen_t addrlen, int16_t* out_code) { |
| SocketAddress fidl_addr; |
| zx_status_t status = fidl_addr.LoadSockAddr(addr, addrlen); |
| if (status != ZX_OK) { |
| return status; |
| } |
| |
| auto response = fidl_addr.WithFIDL( |
| [this](fnet::wire::SocketAddress address) { return client()->Bind(address); }); |
| status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| *out_code = 0; |
| return ZX_OK; |
| } |
| |
| zx_status_t connect(const struct sockaddr* addr, socklen_t addrlen, int16_t* out_code) { |
| // If address is AF_UNSPEC we should call disconnect. |
| if (addr->sa_family == AF_UNSPEC) { |
| auto response = client()->Disconnect(); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| const auto& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| } else { |
| *out_code = 0; |
| } |
| return ZX_OK; |
| } |
| |
| SocketAddress fidl_addr; |
| zx_status_t status = fidl_addr.LoadSockAddr(addr, addrlen); |
| if (status != ZX_OK) { |
| return status; |
| } |
| |
| auto response = fidl_addr.WithFIDL( |
| [this](fnet::wire::SocketAddress address) { return client()->Connect(address); }); |
| status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| *out_code = 0; |
| return ZX_OK; |
| } |
| |
| template <typename R> |
| zx_status_t getname(R&& response, struct sockaddr* addr, socklen_t* addrlen, int16_t* out_code) { |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| if (addrlen == nullptr || (*addrlen != 0 && addr == nullptr)) { |
| *out_code = EFAULT; |
| return ZX_OK; |
| } |
| *out_code = 0; |
| auto const& out = result.value()->addr; |
| *addrlen = fidl_to_sockaddr(out, addr, *addrlen); |
| return ZX_OK; |
| } |
| |
| zx_status_t getsockname(struct sockaddr* addr, socklen_t* addrlen, int16_t* out_code) { |
| return getname(client()->GetSockName(), addr, addrlen, out_code); |
| } |
| |
| zx_status_t getpeername(struct sockaddr* addr, socklen_t* addrlen, int16_t* out_code) { |
| return getname(client()->GetPeerName(), addr, addrlen, out_code); |
| } |
| |
| SockOptResult getsockopt_fidl(int level, int optname, void* optval, socklen_t* optlen) { |
| GetSockOptProcessor proc(optval, optlen); |
| switch (level) { |
| case SOL_SOCKET: |
| return BaseSocket::get_solsocket_sockopt_fidl(optname, optval, optlen); |
| case SOL_IP: |
| switch (optname) { |
| case IP_TTL: |
| return proc.Process(client()->GetIpTtl(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_RECVTTL: |
| return proc.Process(client()->GetIpReceiveTtl(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_MULTICAST_TTL: |
| return proc.Process(client()->GetIpMulticastTtl(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_MULTICAST_IF: |
| return proc.Process(client()->GetIpMulticastInterface(), |
| [](const auto& response) { return response.value; }); |
| case IP_MULTICAST_LOOP: |
| return proc.Process(client()->GetIpMulticastLoopback(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_TOS: |
| return proc.Process(client()->GetIpTypeOfService(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_RECVTOS: |
| return proc.Process(client()->GetIpReceiveTypeOfService(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = true, |
| }; |
| }); |
| case IP_PKTINFO: |
| return proc.Process(client()->GetIpPacketInfo(), |
| [](const auto& response) { return response.value; }); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| case SOL_IPV6: |
| switch (optname) { |
| case IPV6_V6ONLY: |
| return proc.Process(client()->GetIpv6Only(), |
| [](const auto& response) { return response.value; }); |
| case IPV6_TCLASS: |
| return proc.Process(client()->GetIpv6TrafficClass(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_MULTICAST_IF: |
| return proc.Process(client()->GetIpv6MulticastInterface(), [](const auto& response) { |
| return static_cast<uint32_t>(response.value); |
| }); |
| case IPV6_UNICAST_HOPS: |
| return proc.Process(client()->GetIpv6UnicastHops(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_MULTICAST_HOPS: |
| return proc.Process(client()->GetIpv6MulticastHops(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_MULTICAST_LOOP: |
| return proc.Process(client()->GetIpv6MulticastLoopback(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_RECVTCLASS: |
| return proc.Process(client()->GetIpv6ReceiveTrafficClass(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_RECVHOPLIMIT: |
| return proc.Process(client()->GetIpv6ReceiveHopLimit(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| case IPV6_RECVPKTINFO: |
| return proc.Process(client()->GetIpv6ReceivePacketInfo(), [](const auto& response) { |
| return PartialCopy{ |
| .value = response.value, |
| .allow_char = false, |
| }; |
| }); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| case SOL_TCP: |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>>) { |
| switch (optname) { |
| case TCP_NODELAY: |
| return proc.Process(client()->GetTcpNoDelay(), |
| [](const auto& response) { return response.value; }); |
| case TCP_CORK: |
| return proc.Process(client()->GetTcpCork(), |
| [](const auto& response) { return response.value; }); |
| case TCP_QUICKACK: |
| return proc.Process(client()->GetTcpQuickAck(), |
| [](const auto& response) { return response.value; }); |
| case TCP_MAXSEG: |
| return proc.Process(client()->GetTcpMaxSegment(), |
| [](const auto& response) { return response.value_bytes; }); |
| case TCP_KEEPIDLE: |
| return proc.Process(client()->GetTcpKeepAliveIdle(), |
| [](const auto& response) { return response.value_secs; }); |
| case TCP_KEEPINTVL: |
| return proc.Process(client()->GetTcpKeepAliveInterval(), |
| [](const auto& response) { return response.value_secs; }); |
| case TCP_KEEPCNT: |
| return proc.Process(client()->GetTcpKeepAliveCount(), |
| [](const auto& response) { return response.value; }); |
| case TCP_USER_TIMEOUT: |
| return proc.Process(client()->GetTcpUserTimeout(), |
| [](const auto& response) { return response.value_millis; }); |
| case TCP_CONGESTION: |
| return proc.Process(client()->GetTcpCongestion(), [](const auto& response) { |
| switch (response.value) { |
| case fsocket::wire::TcpCongestionControl::kCubic: |
| return TruncatingStringView( |
| fidl::StringView::FromExternal(kCcCubic, sizeof(kCcCubic))); |
| case fsocket::wire::TcpCongestionControl::kReno: |
| return TruncatingStringView( |
| fidl::StringView::FromExternal(kCcReno, sizeof(kCcReno))); |
| } |
| }); |
| case TCP_DEFER_ACCEPT: |
| return proc.Process(client()->GetTcpDeferAccept(), |
| [](const auto& response) { return response.value_secs; }); |
| case TCP_INFO: |
| return proc.Process( |
| client()->GetTcpInfo(), |
| [](const auto& response) -> const auto& { return response.info; }); |
| case TCP_SYNCNT: |
| return proc.Process(client()->GetTcpSynCount(), |
| [](const auto& response) { return response.value; }); |
| case TCP_WINDOW_CLAMP: |
| return proc.Process(client()->GetTcpWindowClamp(), |
| [](const auto& response) { return response.value; }); |
| case TCP_LINGER2: |
| return proc.Process(client()->GetTcpLinger(), |
| [](const auto& response) -> const fsocket::wire::OptionalUint32& { |
| return response.value_secs; |
| }); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| } else { |
| __FALLTHROUGH; |
| } |
| default: |
| return SockOptResult::Errno(EPROTONOSUPPORT); |
| } |
| } |
| |
| SockOptResult setsockopt_fidl(int level, int optname, const void* optval, socklen_t optlen) { |
| SetSockOptProcessor proc(optval, optlen); |
| switch (level) { |
| case SOL_SOCKET: |
| return BaseSocket::set_solsocket_sockopt_fidl(optname, optval, optlen); |
| case SOL_IP: |
| switch (optname) { |
| case IP_MULTICAST_TTL: |
| return proc.Process<OptionalUint8CharAllowed>([this](OptionalUint8CharAllowed value) { |
| return client()->SetIpMulticastTtl(value.inner); |
| }); |
| case IP_ADD_MEMBERSHIP: { |
| return proc.Process<fsocket::wire::IpMulticastMembership>( |
| [this](fsocket::wire::IpMulticastMembership value) { |
| return client()->AddIpMembership(value); |
| }); |
| } |
| case IP_DROP_MEMBERSHIP: |
| return proc.Process<fsocket::wire::IpMulticastMembership>( |
| [this](fsocket::wire::IpMulticastMembership value) { |
| return client()->DropIpMembership(value); |
| }); |
| case IP_MULTICAST_IF: { |
| if (optlen == sizeof(struct in_addr)) { |
| return proc.Process<struct in_addr>([this](struct in_addr value) { |
| fnet::wire::Ipv4Address addr; |
| static_assert(sizeof(addr.addr) == sizeof(value.s_addr)); |
| memcpy(addr.addr.data(), &value.s_addr, sizeof(value.s_addr)); |
| return client()->SetIpMulticastInterface(0, addr); |
| }); |
| } |
| return proc.Process<fsocket::wire::IpMulticastMembership>( |
| [this](fsocket::wire::IpMulticastMembership value) { |
| return client()->SetIpMulticastInterface(value.iface, value.local_addr); |
| }); |
| } |
| case IP_MULTICAST_LOOP: |
| return proc.Process<IntOrChar>([this](IntOrChar value) { |
| return client()->SetIpMulticastLoopback(value.value != 0); |
| }); |
| case IP_TTL: |
| return proc.Process<OptionalUint8CharAllowed>( |
| [this](OptionalUint8CharAllowed value) { return client()->SetIpTtl(value.inner); }); |
| case IP_RECVTTL: |
| return proc.Process<IntOrChar>( |
| [this](IntOrChar value) { return client()->SetIpReceiveTtl(value.value != 0); }); |
| case IP_TOS: |
| if (optlen == 0) { |
| return SockOptResult::Ok(); |
| } |
| return proc.Process<IntOrChar>([this](IntOrChar value) { |
| return client()->SetIpTypeOfService(static_cast<uint8_t>(value.value)); |
| }); |
| case IP_RECVTOS: |
| return proc.Process<IntOrChar>([this](IntOrChar value) { |
| return client()->SetIpReceiveTypeOfService(value.value != 0); |
| }); |
| case IP_PKTINFO: |
| return proc.Process<IntOrChar>( |
| [this](IntOrChar value) { return client()->SetIpPacketInfo(value.value != 0); }); |
| case MCAST_JOIN_GROUP: |
| return SockOptResult::Errno(ENOTSUP); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| case SOL_IPV6: |
| switch (optname) { |
| case IPV6_V6ONLY: |
| return proc.Process<bool>([this](bool value) { return client()->SetIpv6Only(value); }); |
| case IPV6_ADD_MEMBERSHIP: |
| return proc.Process<fsocket::wire::Ipv6MulticastMembership>( |
| [this](fsocket::wire::Ipv6MulticastMembership value) { |
| return client()->AddIpv6Membership(value); |
| }); |
| case IPV6_DROP_MEMBERSHIP: |
| return proc.Process<fsocket::wire::Ipv6MulticastMembership>( |
| [this](fsocket::wire::Ipv6MulticastMembership value) { |
| return client()->DropIpv6Membership(value); |
| }); |
| case IPV6_MULTICAST_IF: |
| return proc.Process<IntOrChar>([this](IntOrChar value) { |
| return client()->SetIpv6MulticastInterface(value.value); |
| }); |
| case IPV6_UNICAST_HOPS: |
| return proc.Process<fsocket::wire::OptionalUint8>( |
| [this](fsocket::wire::OptionalUint8 value) { |
| return client()->SetIpv6UnicastHops(value); |
| }); |
| case IPV6_MULTICAST_HOPS: |
| return proc.Process<fsocket::wire::OptionalUint8>( |
| [this](fsocket::wire::OptionalUint8 value) { |
| return client()->SetIpv6MulticastHops(value); |
| }); |
| case IPV6_MULTICAST_LOOP: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetIpv6MulticastLoopback(value); }); |
| case IPV6_TCLASS: |
| return proc.Process<fsocket::wire::OptionalUint8>( |
| [this](fsocket::wire::OptionalUint8 value) { |
| return client()->SetIpv6TrafficClass(value); |
| }); |
| case IPV6_RECVTCLASS: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetIpv6ReceiveTrafficClass(value); }); |
| case IPV6_RECVHOPLIMIT: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetIpv6ReceiveHopLimit(value); }); |
| case IPV6_RECVPKTINFO: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetIpv6ReceivePacketInfo(value); }); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| case SOL_TCP: |
| if constexpr (std::is_same_v<T, fidl::WireSyncClient<fsocket::StreamSocket>>) { |
| switch (optname) { |
| case TCP_NODELAY: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetTcpNoDelay(value); }); |
| case TCP_CORK: |
| return proc.Process<bool>([this](bool value) { return client()->SetTcpCork(value); }); |
| case TCP_QUICKACK: |
| return proc.Process<bool>( |
| [this](bool value) { return client()->SetTcpQuickAck(value); }); |
| case TCP_MAXSEG: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpMaxSegment(value); }); |
| case TCP_KEEPIDLE: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpKeepAliveIdle(value); }); |
| case TCP_KEEPINTVL: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpKeepAliveInterval(value); }); |
| case TCP_KEEPCNT: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpKeepAliveCount(value); }); |
| case TCP_USER_TIMEOUT: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpUserTimeout(value); }); |
| case TCP_CONGESTION: |
| return proc.Process<fsocket::wire::TcpCongestionControl>( |
| [this](fsocket::wire::TcpCongestionControl value) { |
| return client()->SetTcpCongestion(value); |
| }); |
| case TCP_DEFER_ACCEPT: |
| return proc.Process<int32_t>([this](int32_t value) { |
| if (value < 0) { |
| value = 0; |
| } |
| return client()->SetTcpDeferAccept(value); |
| }); |
| case TCP_SYNCNT: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpSynCount(value); }); |
| case TCP_WINDOW_CLAMP: |
| return proc.Process<uint32_t>( |
| [this](uint32_t value) { return client()->SetTcpWindowClamp(value); }); |
| case TCP_LINGER2: |
| return proc.Process<int32_t>([this](int32_t value) { |
| fsocket::wire::OptionalUint32 opt; |
| if (value < 0) { |
| opt = fsocket::wire::OptionalUint32::WithUnset({}); |
| } else { |
| opt = fsocket::wire::OptionalUint32::WithValue(static_cast<uint32_t>(value)); |
| } |
| return client()->SetTcpLinger(opt); |
| }); |
| default: |
| return SockOptResult::Errno(ENOPROTOOPT); |
| } |
| } else { |
| __FALLTHROUGH; |
| } |
| default: |
| return SockOptResult::Errno(EPROTONOSUPPORT); |
| } |
| } |
| |
| zx_status_t shutdown(int how, int16_t* out_code) { |
| using fsocket::wire::ShutdownMode; |
| ShutdownMode mode; |
| switch (how) { |
| case SHUT_RD: |
| mode = ShutdownMode::kRead; |
| break; |
| case SHUT_WR: |
| mode = ShutdownMode::kWrite; |
| break; |
| case SHUT_RDWR: |
| mode = ShutdownMode::kRead | ShutdownMode::kWrite; |
| break; |
| default: |
| return ZX_ERR_INVALID_ARGS; |
| } |
| |
| auto response = client()->Shutdown(mode); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| *out_code = 0; |
| return ZX_OK; |
| } |
| }; |
| |
| // Prevent divergence in flag bitmasks between libc and fuchsia.posix.socket FIDL library. |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kUp) == IFF_UP); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kBroadcast) == IFF_BROADCAST); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kDebug) == IFF_DEBUG); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kLoopback) == IFF_LOOPBACK); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kPointtopoint) == |
| IFF_POINTOPOINT); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kNotrailers) == IFF_NOTRAILERS); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kRunning) == IFF_RUNNING); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kNoarp) == IFF_NOARP); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kPromisc) == IFF_PROMISC); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kAllmulti) == IFF_ALLMULTI); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kLeader) == IFF_MASTER); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kFollower) == IFF_SLAVE); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kMulticast) == IFF_MULTICAST); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kPortsel) == IFF_PORTSEL); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kAutomedia) == IFF_AUTOMEDIA); |
| static_assert(static_cast<uint16_t>(fsocket::wire::InterfaceFlags::kDynamic) == IFF_DYNAMIC); |
| |
| } // namespace |
| |
| namespace fdio_internal { |
| |
| struct SynchronousDatagramSocket; |
| struct RawSocket; |
| struct PacketSocket; |
| struct StreamSocket; |
| |
| template <typename T, |
| typename = std::enable_if_t< |
| std::is_same_v<T, SynchronousDatagramSocket> || std::is_same_v<T, RawSocket> || |
| std::is_same_v<T, PacketSocket> || std::is_same_v<T, StreamSocket>>> |
| struct base_socket : public zxio { |
| Errno posix_ioctl(int req, va_list va) final { |
| switch (req) { |
| case SIOCGIFNAME: { |
| auto& provider = get_client<fsocket::Provider>(); |
| if (provider.is_error()) { |
| return Errno(fdio_status_to_errno(provider.error_value())); |
| } |
| struct ifreq* ifr = va_arg(va, struct ifreq*); |
| auto response = provider->InterfaceIndexToName(static_cast<uint64_t>(ifr->ifr_ifindex)); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return Errno(fdio_status_to_errno(status)); |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| if (result.error_value() == ZX_ERR_NOT_FOUND) { |
| return Errno(ENODEV); |
| } |
| return Errno(fdio_status_to_errno(result.error_value())); |
| } |
| auto const& if_name = result.value()->name; |
| const size_t len = std::min(if_name.size(), std::size(ifr->ifr_name)); |
| auto it = std::copy_n(if_name.begin(), len, std::begin(ifr->ifr_name)); |
| if (it != std::end(ifr->ifr_name)) { |
| *it = 0; |
| } |
| return Errno(Errno::Ok); |
| } |
| case SIOCGIFINDEX: { |
| auto& provider = get_client<fsocket::Provider>(); |
| if (provider.is_error()) { |
| return Errno(fdio_status_to_errno(provider.error_value())); |
| } |
| struct ifreq* ifr = va_arg(va, struct ifreq*); |
| fidl::StringView name(ifr->ifr_name, strnlen(ifr->ifr_name, sizeof(ifr->ifr_name) - 1)); |
| auto response = provider->InterfaceNameToIndex(name); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| if (status == ZX_ERR_INVALID_ARGS) { |
| // FIDL calls will return ZX_ERR_INVALID_ARGS if the passed string |
| // (`name` in this case) fails UTF-8 validation. |
| return Errno(ENODEV); |
| } |
| return Errno(fdio_status_to_errno(status)); |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| if (result.error_value() == ZX_ERR_NOT_FOUND) { |
| return Errno(ENODEV); |
| } |
| return Errno(fdio_status_to_errno(result.error_value())); |
| } |
| ifr->ifr_ifindex = static_cast<int>(result.value()->index); |
| return Errno(Errno::Ok); |
| } |
| case SIOCGIFFLAGS: { |
| auto& provider = get_client<fsocket::Provider>(); |
| if (provider.is_error()) { |
| return Errno(fdio_status_to_errno(provider.error_value())); |
| } |
| struct ifreq* ifr = va_arg(va, struct ifreq*); |
| fidl::StringView name(ifr->ifr_name, strnlen(ifr->ifr_name, sizeof(ifr->ifr_name) - 1)); |
| auto response = provider->InterfaceNameToFlags(name); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| if (status == ZX_ERR_INVALID_ARGS) { |
| // FIDL calls will return ZX_ERR_INVALID_ARGS if the passed string |
| // (`name` in this case) fails UTF-8 validation. |
| return Errno(ENODEV); |
| } |
| return Errno(fdio_status_to_errno(status)); |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| if (result.error_value() == ZX_ERR_NOT_FOUND) { |
| return Errno(ENODEV); |
| } |
| return Errno(fdio_status_to_errno(result.error_value())); |
| } |
| ifr->ifr_flags = |
| static_cast<uint16_t>(result.value()->flags); // NOLINT(bugprone-narrowing-conversions) |
| return Errno(Errno::Ok); |
| } |
| case SIOCGIFCONF: { |
| struct ifconf* ifc_ptr = va_arg(va, struct ifconf*); |
| if (ifc_ptr == nullptr) { |
| return Errno(EFAULT); |
| } |
| struct ifconf& ifc = *ifc_ptr; |
| |
| auto& provider = get_client<fsocket::Provider>(); |
| if (provider.is_error()) { |
| return Errno(fdio_status_to_errno(provider.error_value())); |
| } |
| auto response = provider->GetInterfaceAddresses(); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return Errno(fdio_status_to_errno(status)); |
| } |
| const auto& interfaces = response.value().interfaces; |
| |
| // If `ifc_req` is NULL, return the necessary buffer size in bytes for |
| // receiving all available addresses in `ifc_len`. |
| // |
| // This allows the caller to determine the necessary buffer size |
| // beforehand, and is the documented manual behavior. |
| // See: https://man7.org/linux/man-pages/man7/netdevice.7.html |
| if (ifc.ifc_req == nullptr) { |
| int len = 0; |
| for (const auto& iface : interfaces) { |
| for (const auto& address : iface.addresses()) { |
| if (address.addr.Which() == fnet::wire::IpAddress::Tag::kIpv4) { |
| len += sizeof(struct ifreq); |
| } |
| } |
| } |
| ifc.ifc_len = len; |
| return Errno(Errno::Ok); |
| } |
| |
| struct ifreq* ifr = ifc.ifc_req; |
| const auto buffer_full = [&] { |
| return ifr + 1 > ifc.ifc_req + ifc.ifc_len / sizeof(struct ifreq); |
| }; |
| for (const auto& iface : interfaces) { |
| // Don't write past the caller-allocated buffer. |
| // C++ doesn't support break labels, so we check this in both the inner |
| // and outer loops. |
| if (buffer_full()) { |
| break; |
| } |
| // This should not happen, and would indicate a protocol error with |
| // fuchsia.posix.socket/Provider.GetInterfaceAddresses. |
| if (!iface.has_name() || !iface.has_addresses()) { |
| continue; |
| } |
| |
| const auto& if_name = iface.name(); |
| for (const auto& address : iface.addresses()) { |
| // Don't write past the caller-allocated buffer. |
| if (buffer_full()) { |
| break; |
| } |
| // SIOCGIFCONF only returns interface addresses of the AF_INET (IPv4) |
| // family for compatibility; this is the behavior documented in the |
| // manual. See: https://man7.org/linux/man-pages/man7/netdevice.7.html |
| const auto& addr = address.addr; |
| if (addr.Which() != fnet::wire::IpAddress::Tag::kIpv4) { |
| continue; |
| } |
| |
| // Write interface name. |
| const size_t len = std::min(if_name.size(), std::size(ifr->ifr_name)); |
| auto it = std::copy_n(if_name.begin(), len, std::begin(ifr->ifr_name)); |
| if (it != std::end(ifr->ifr_name)) { |
| *it = 0; |
| } |
| |
| // Write interface address. |
| auto& s = *reinterpret_cast<struct sockaddr_in*>(&ifr->ifr_addr); |
| const auto& ipv4 = addr.ipv4(); |
| s.sin_family = AF_INET; |
| s.sin_port = 0; |
| static_assert(sizeof(s.sin_addr) == sizeof(ipv4.addr)); |
| memcpy(&s.sin_addr, ipv4.addr.data(), sizeof(ipv4.addr)); |
| |
| ifr++; |
| } |
| } |
| ifc.ifc_len = static_cast<int>((ifr - ifc.ifc_req) * sizeof(struct ifreq)); |
| return Errno(Errno::Ok); |
| } |
| default: |
| return zxio::posix_ioctl(req, va); |
| } |
| } |
| |
| protected: |
| virtual fidl::WireSyncClient<typename T::FidlProtocol>& GetClient() = 0; |
| }; |
| |
| void recvmsg_populate_socketaddress(const fnet::wire::SocketAddress& fidl, void* addr, |
| socklen_t& addr_len) { |
| // Result address has invalid tag when it's not provided by the server (when the address |
| // is not requested). |
| // TODO(https://fxbug.dev/58503): Use better representation of nullable union when available. |
| if (fidl.has_invalid_tag()) { |
| return; |
| } |
| |
| addr_len = fidl_to_sockaddr(fidl, addr, addr_len); |
| } |
| |
| struct StreamSocket { |
| using FidlProtocol = fsocket::StreamSocket; |
| }; |
| |
| struct SynchronousDatagramSocket { |
| using FidlProtocol = fsocket::SynchronousDatagramSocket; |
| using FidlSockAddr = SocketAddress; |
| using FidlSendControlData = fsocket::wire::DatagramSocketSendControlData; |
| using zxio_type = zxio_synchronous_datagram_socket_t; |
| |
| static void recvmsg_populate_msgname( |
| const fsocket::wire::SynchronousDatagramSocketRecvMsgResponse& response, void* addr, |
| socklen_t& addr_len) { |
| recvmsg_populate_socketaddress(response.addr, addr, addr_len); |
| } |
| |
| static void handle_sendmsg_response( |
| const fsocket::wire::SynchronousDatagramSocketSendMsgResponse& response, |
| ssize_t expected_len) { |
| // TODO(https://fxbug.dev/82346): Drop len from the response as SendMsg does |
| // does not perform partial writes. |
| ZX_DEBUG_ASSERT_MSG(response.len == expected_len, "got SendMsg(...) = %ld, want = %ld", |
| response.len, expected_len); |
| } |
| }; |
| |
| struct RawSocket { |
| using FidlProtocol = frawsocket::Socket; |
| using FidlSockAddr = SocketAddress; |
| using FidlSendControlData = fsocket::wire::NetworkSocketSendControlData; |
| using zxio_type = zxio_raw_socket_t; |
| |
| static void recvmsg_populate_msgname(const frawsocket::wire::SocketRecvMsgResponse& response, |
| void* addr, socklen_t& addr_len) { |
| recvmsg_populate_socketaddress(response.addr, addr, addr_len); |
| } |
| |
| static void handle_sendmsg_response(const frawsocket::wire::SocketSendMsgResponse& response, |
| ssize_t expected_len) { |
| // TODO(https://fxbug.dev/82346): Drop this method once DatagramSocket.SendMsg |
| // no longer returns a length field. |
| } |
| }; |
| |
| struct PacketSocket { |
| using FidlProtocol = fpacketsocket::Socket; |
| using FidlSockAddr = PacketInfo; |
| using FidlSendControlData = fpacketsocket::wire::SendControlData; |
| using zxio_type = zxio_packet_socket_t; |
| |
| static void recvmsg_populate_msgname(const fpacketsocket::wire::SocketRecvMsgResponse& response, |
| void* addr, socklen_t& addr_len) { |
| fidl::ObjectView view = response.packet_info; |
| if (!view) { |
| // The packet info field is not provided by the server (when it is not requested). |
| return; |
| } |
| |
| const fpacketsocket::wire::RecvPacketInfo& info = *view; |
| |
| sockaddr_ll sll = { |
| .sll_family = AF_PACKET, |
| .sll_protocol = htons(info.packet_info.protocol), |
| .sll_ifindex = static_cast<int>(info.packet_info.interface_id), |
| .sll_hatype = fidl_hwtype_to_arphrd(info.interface_type), |
| .sll_pkttype = fidl_pkttype_to_pkttype(info.packet_type), |
| }; |
| populate_from_fidl_hwaddr(info.packet_info.addr, sll); |
| memcpy(addr, &sll, std::min(sizeof(sll), static_cast<size_t>(addr_len))); |
| addr_len = sizeof(sll); |
| } |
| |
| static void handle_sendmsg_response(const fpacketsocket::wire::SocketSendMsgResponse& response, |
| ssize_t expected_len) { |
| // TODO(https://fxbug.dev/82346): Drop this method once DatagramSocket.SendMsg |
| // no longer returns a length field. |
| } |
| }; |
| |
| template <typename R, typename = int> |
| struct FitxResultHasValue : std::false_type {}; |
| template <typename R> |
| struct FitxResultHasValue<R, decltype(&R::value, 0)> : std::true_type {}; |
| template <typename T, typename R> |
| typename std::enable_if<FitxResultHasValue<R>::value>::type HandleSendMsgResponse(const R& result, |
| size_t total) { |
| T::handle_sendmsg_response(*result->value(), total); |
| } |
| template <typename T, typename R> |
| typename std::enable_if<!FitxResultHasValue<T>::value>::type HandleSendMsgResponse(const R& result, |
| size_t total) {} |
| |
| template <typename T, typename = std::enable_if_t<std::is_same_v<T, SynchronousDatagramSocket> || |
| std::is_same_v<T, RawSocket> || |
| std::is_same_v<T, PacketSocket>>> |
| // inheritance is virtual to avoid multiple copies of `base_socket<T>` when derived classes |
| // inherit from `socket_with_event` and `network_socket`. |
| struct socket_with_event : virtual public base_socket<T> { |
| static constexpr zx_signals_t kSignalIncoming = ZX_USER_SIGNAL_0; |
| static constexpr zx_signals_t kSignalOutgoing = ZX_USER_SIGNAL_1; |
| static constexpr zx_signals_t kSignalError = ZX_USER_SIGNAL_2; |
| static constexpr zx_signals_t kSignalShutdownRead = ZX_USER_SIGNAL_4; |
| static constexpr zx_signals_t kSignalShutdownWrite = ZX_USER_SIGNAL_5; |
| |
| void wait_begin(uint32_t events, zx_handle_t* handle, zx_signals_t* out_signals) override { |
| *handle = zxio_socket_with_event().event.get(); |
| |
| zx_signals_t signals = ZX_EVENTPAIR_PEER_CLOSED | kSignalError; |
| if (events & POLLIN) { |
| signals |= kSignalIncoming | kSignalShutdownRead; |
| } |
| if (events & POLLOUT) { |
| signals |= kSignalOutgoing | kSignalShutdownWrite; |
| } |
| if (events & POLLRDHUP) { |
| signals |= kSignalShutdownRead; |
| } |
| *out_signals = signals; |
| } |
| |
| void wait_end(zx_signals_t signals, uint32_t* out_events) override { |
| uint32_t events = 0; |
| if (signals & (ZX_EVENTPAIR_PEER_CLOSED | kSignalIncoming | kSignalShutdownRead)) { |
| events |= POLLIN; |
| } |
| if (signals & (ZX_EVENTPAIR_PEER_CLOSED | kSignalOutgoing | kSignalShutdownWrite)) { |
| events |= POLLOUT; |
| } |
| if (signals & (ZX_EVENTPAIR_PEER_CLOSED | kSignalError)) { |
| events |= POLLERR; |
| } |
| if (signals & (ZX_EVENTPAIR_PEER_CLOSED | kSignalShutdownRead)) { |
| events |= POLLRDHUP; |
| } |
| *out_events = events; |
| } |
| |
| zx_status_t recvmsg(struct msghdr* msg, int flags, size_t* out_actual, |
| int16_t* out_code) override { |
| size_t datalen = 0; |
| for (int i = 0; i < msg->msg_iovlen; ++i) { |
| datalen += msg->msg_iov[i].iov_len; |
| } |
| |
| bool want_addr = msg->msg_namelen != 0 && msg->msg_name != nullptr; |
| bool want_cmsg = msg->msg_controllen != 0 && msg->msg_control != nullptr; |
| auto response = GetClient()->RecvMsg(want_addr, static_cast<uint32_t>(datalen), want_cmsg, |
| to_recvmsg_flags(flags)); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| *out_code = 0; |
| |
| T::recvmsg_populate_msgname(*result.value(), msg->msg_name, msg->msg_namelen); |
| |
| { |
| auto const& out = result.value()->data; |
| |
| const uint8_t* data = out.begin(); |
| size_t remaining = out.count(); |
| for (int i = 0; remaining != 0 && i < msg->msg_iovlen; ++i) { |
| iovec const& iov = msg->msg_iov[i]; |
| if (iov.iov_base != nullptr) { |
| size_t actual = std::min(iov.iov_len, remaining); |
| memcpy(iov.iov_base, data, actual); |
| data += actual; |
| remaining -= actual; |
| } else if (iov.iov_len != 0) { |
| *out_code = EFAULT; |
| return ZX_OK; |
| } |
| } |
| if (result.value()->truncated != 0) { |
| msg->msg_flags |= MSG_TRUNC; |
| } else { |
| msg->msg_flags &= ~MSG_TRUNC; |
| } |
| size_t actual = out.count() - remaining; |
| if ((flags & MSG_TRUNC) != 0) { |
| actual += result.value()->truncated; |
| } |
| *out_actual = actual; |
| } |
| |
| if (want_cmsg) { |
| FidlControlDataProcessor proc(msg->msg_control, msg->msg_controllen); |
| msg->msg_controllen = proc.Store(result.value()->control); |
| } else { |
| msg->msg_controllen = 0; |
| } |
| |
| return ZX_OK; |
| } |
| |
| zx_status_t sendmsg(const struct msghdr* msg, int flags, size_t* out_actual, |
| int16_t* out_code) override { |
| typename T::FidlSockAddr addr; |
| // Attempt to load socket address if either name or namelen is set. |
| // If only one is set, it'll result in INVALID_ARGS. |
| if (msg->msg_namelen != 0 || msg->msg_name != nullptr) { |
| zx_status_t status = |
| addr.LoadSockAddr(static_cast<struct sockaddr*>(msg->msg_name), msg->msg_namelen); |
| if (status != ZX_OK) { |
| return status; |
| } |
| } |
| |
| size_t total = 0; |
| for (int i = 0; i < msg->msg_iovlen; ++i) { |
| iovec const& iov = msg->msg_iov[i]; |
| if (iov.iov_base == nullptr && iov.iov_len != 0) { |
| *out_code = EFAULT; |
| return ZX_OK; |
| } |
| total += iov.iov_len; |
| } |
| |
| fidl::Arena allocator; |
| fitx::result cmsg_result = ParseControlMessages<typename T::FidlSendControlData>( |
| msg->msg_control, msg->msg_controllen, allocator); |
| if (cmsg_result.is_error()) { |
| *out_code = cmsg_result.error_value(); |
| return ZX_OK; |
| } |
| const typename T::FidlSendControlData& cdata = cmsg_result.value(); |
| |
| std::vector<uint8_t> data; |
| auto vec = fidl::VectorView<uint8_t>(); |
| switch (msg->msg_iovlen) { |
| case 0: { |
| break; |
| } |
| case 1: { |
| iovec const& iov = *msg->msg_iov; |
| vec = fidl::VectorView<uint8_t>::FromExternal(static_cast<uint8_t*>(iov.iov_base), |
| iov.iov_len); |
| break; |
| } |
| default: { |
| // TODO(https://fxbug.dev/67928): avoid this copy. |
| data.reserve(total); |
| for (int i = 0; i < msg->msg_iovlen; ++i) { |
| iovec const& iov = msg->msg_iov[i]; |
| std::copy_n(static_cast<const uint8_t*>(iov.iov_base), iov.iov_len, |
| std::back_inserter(data)); |
| } |
| vec = fidl::VectorView<uint8_t>::FromExternal(data); |
| } |
| } |
| |
| // TODO(https://fxbug.dev/58503): Use better representation of nullable union when |
| // available. Currently just using a default-initialized union with an invalid tag. |
| auto response = addr.WithFIDL([&](auto address) { |
| return GetClient()->SendMsg(address, vec, cdata, to_sendmsg_flags(flags)); |
| }); |
| zx_status_t status = response.status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| auto const& result = response.value(); |
| if (result.is_error()) { |
| *out_code = static_cast<int16_t>(result.error_value()); |
| return ZX_OK; |
| } |
| HandleSendMsgResponse<T, decltype(result)>(result, total); |
| |
| *out_code = 0; |
| // SendMsg does not perform partial writes. |
| *out_actual = total; |
| return ZX_OK; |
| } |
| |
| protected: |
| friend class fbl::internal::MakeRefCountedHelper<socket_with_event<T>>; |
| friend class fbl::RefPtr<socket_with_event<T>>; |
| |
| socket_with_event<T>() = default; |
| ~socket_with_event<T>() override = default; |
| |
| fidl::WireSyncClient<typename T::FidlProtocol>& GetClient() override { |
| return zxio_socket_with_event().client; |
| } |
| |
| typename T::zxio_type& zxio_socket_with_event() { |
| return *reinterpret_cast<typename T::zxio_type*>(&base_socket<T>::zxio_storage().io); |
| } |
| }; |
| |
| template <typename T, typename = std::enable_if_t<std::is_same_v<T, SynchronousDatagramSocket> || |
| std::is_same_v<T, StreamSocket> || |
| std::is_same_v<T, RawSocket>>> |
| // inheritance is virtual to avoid multiple copies of `base_socket<T>` when derived classes |
| // inherit from `network_socket` and `socket_with_event`. |
| struct network_socket : virtual public base_socket<T> { |
| using base_socket<T>::GetClient; |
| zx_status_t bind(const struct sockaddr* addr, socklen_t addrlen, int16_t* out_code) override { |
| return BaseNetworkSocket(GetClient()).bind(addr, addrlen, out_code); |
| } |
| |
| zx_status_t connect(const struct sockaddr* addr, socklen_t addrlen, int16_t* out_code) override { |
| return BaseNetworkSocket(GetClient()).connect(addr, addrlen, out_code); |
| } |
| |
| zx_status_t getsockname(struct sockaddr* addr, socklen_t* addrlen, int16_t* out_code) override { |
| return BaseNetworkSocket(GetClient()).getsockname(addr, addrlen, out_code); |
| } |
| |
| zx_status_t getpeername(struct sockaddr* addr, socklen_t* addrlen, int16_t* out_code) override { |
| return BaseNetworkSocket(GetClient()).getpeername(addr, addrlen, out_code); |
| } |
| |
| zx_status_t getsockopt(int level, int optname, void* optval, socklen_t* optlen, |
| int16_t* out_code) override { |
| SockOptResult result = |
| BaseNetworkSocket(GetClient()).getsockopt_fidl(level, optname, optval, optlen); |
| *out_code = result.err; |
| return result.status; |
| } |
| |
| zx_status_t setsockopt(int level, int optname, const void* optval, socklen_t optlen, |
| int16_t* out_code) override { |
| SockOptResult result = |
| BaseNetworkSocket(GetClient()).setsockopt_fidl(level, optname, optval, optlen); |
| *out_code = result.err; |
| return result.status; |
| } |
| |
| zx_status_t shutdown(int how, int16_t* out_code) override { |
| return BaseNetworkSocket(GetClient()).shutdown(how, out_code); |
| } |
| }; |
| |
| template <> |
| zx_status_t network_socket<RawSocket>::getsockopt(int level, int optname, void* optval, |
| socklen_t* optlen, int16_t* out_code) { |
| SockOptResult result = [&]() { |
| GetSockOptProcessor proc(optval, optlen); |
| switch (level) { |
| case SOL_ICMPV6: |
| switch (optname) { |
| case ICMP6_FILTER: |
| return proc.Process(GetClient()->GetIcmpv6Filter(), |
| [](const auto& response) { return response.filter; }); |
| } |
| break; |
| case SOL_IPV6: |
| switch (optname) { |
| case IPV6_CHECKSUM: |
| return proc.Process(GetClient()->GetIpv6Checksum(), [](const auto& response) { |
| switch (response.config.Which()) { |
| case frawsocket::wire::Ipv6ChecksumConfiguration::Tag::kDisabled: |
| return -1; |
| case frawsocket::wire::Ipv6ChecksumConfiguration::Tag::kOffset: |
| return response.config.offset(); |
| }; |
| }); |
| } |
| break; |
| case SOL_IP: |
| switch (optname) { |
| case IP_HDRINCL: |
| return proc.Process(GetClient()->GetIpHeaderIncluded(), |
| [](const auto& response) { return response.value; }); |
| } |
| break; |
| } |
| return BaseNetworkSocket(GetClient()).getsockopt_fidl(level, optname, optval, optlen); |
| }(); |
| *out_code = result.err; |
| return result.status; |
| } |
| |
| template <> |
| zx_status_t network_socket<RawSocket>::setsockopt(int level, int optname, const void* optval, |
| socklen_t optlen, int16_t* out_code) { |
| SockOptResult result = [&]() { |
| SetSockOptProcessor proc(optval, optlen); |
| |
| switch (level) { |
| case SOL_ICMPV6: |
| switch (optname) { |
| case ICMP6_FILTER: |
| return proc.Process<frawsocket::wire::Icmpv6Filter>( |
| [this](frawsocket::wire::Icmpv6Filter value) { |
| return GetClient()->SetIcmpv6Filter(value); |
| }); |
| } |
| break; |
| case SOL_IPV6: |
| switch (optname) { |
| case IPV6_CHECKSUM: |
| return proc.Process<int32_t>([this](int32_t value) { |
| frawsocket::wire::Ipv6ChecksumConfiguration config; |
| |
| if (value < 0) { |
| config = frawsocket::wire::Ipv6ChecksumConfiguration::WithDisabled( |
| frawsocket::wire::Empty{}); |
| } else { |
| config = frawsocket::wire::Ipv6ChecksumConfiguration::WithOffset(value); |
| } |
| |
| return GetClient()->SetIpv6Checksum(config); |
| }); |
| } |
| break; |
| case SOL_IP: |
| switch (optname) { |
| case IP_HDRINCL: |
| return proc.Process<bool>( |
| [this](bool value) { return GetClient()->SetIpHeaderIncluded(value); }); |
| } |
| break; |
| } |
| return BaseNetworkSocket(GetClient()).setsockopt_fidl(level, optname, optval, optlen); |
| }(); |
| *out_code = result.err; |
| return result.status; |
| } |
| |
| template <typename T, typename = std::enable_if_t<std::is_same_v<T, SynchronousDatagramSocket> || |
| std::is_same_v<T, RawSocket>>> |
| struct network_socket_with_event : public socket_with_event<T>, public network_socket<T> { |
| protected: |
| friend class fbl::internal::MakeRefCountedHelper<network_socket_with_event<T>>; |
| friend class fbl::RefPtr<network_socket_with_event<T>>; |
| |
| network_socket_with_event<T>() = default; |
| ~network_socket_with_event<T>() override = default; |
| }; |
| |
| using synchronous_datagram_socket = network_socket_with_event<SynchronousDatagramSocket>; |
| using raw_socket = network_socket_with_event<RawSocket>; |
| |
| } // namespace fdio_internal |
| |
| zx::status<fdio_ptr> fdio_synchronous_datagram_socket_create( |
| zx::eventpair event, fidl::ClientEnd<fsocket::SynchronousDatagramSocket> client) { |
| fdio_ptr io = fbl::MakeRefCounted<fdio_internal::synchronous_datagram_socket>(); |
| if (io == nullptr) { |
| return zx::error(ZX_ERR_NO_MEMORY); |
| } |
| zx_status_t status = zxio::CreateSynchronousDatagramSocket(&io->zxio_storage(), std::move(event), |
| std::move(client)); |
| if (status != ZX_OK) { |
| return zx::error(status); |
| } |
| return zx::ok(io); |
| } |
| |
| zx::status<fdio_ptr> fdio_raw_socket_create(zx::eventpair event, |
| fidl::ClientEnd<frawsocket::Socket> client) { |
| fdio_ptr io = fbl::MakeRefCounted<fdio_internal::raw_socket>(); |
| if (io == nullptr) { |
| return zx::error(ZX_ERR_NO_MEMORY); |
| } |
| zx_status_t status = |
| zxio::CreateRawSocket(&io->zxio_storage(), std::move(event), std::move(client)); |
| if (status != ZX_OK) { |
| return zx::error(status); |
| } |
| return zx::ok(io); |
| } |
| |
| static zxio_stream_socket_t& zxio_stream_socket(zxio_t* io) { |
| return *reinterpret_cast<zxio_stream_socket_t*>(io); |
| } |
| |
| namespace fdio_internal { |
| |
| struct stream_socket : public network_socket<StreamSocket> { |
| static constexpr zx_signals_t kSignalIncoming = ZX_USER_SIGNAL_0; |
| static constexpr zx_signals_t kSignalConnected = ZX_USER_SIGNAL_3; |
| |
| enum class State { |
| kUnconnected, |
| kListening, |
| kConnecting, |
| kConnected, |
| }; |
| |
| void wait_begin(uint32_t events, zx_handle_t* handle, zx_signals_t* out_signals) override { |
| zxio_signals_t signals = ZXIO_SIGNAL_PEER_CLOSED; |
| |
| auto [state, has_error] = GetState(); |
| switch (state) { |
| case State::kUnconnected: |
| // Stream sockets which are non-listening or unconnected do not have a potential peer |
| // to generate any waitable signals, skip signal waiting and notify the caller of the |
| // same. |
| |