|  | /* | 
|  | * Copyright (C) 2017 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "perfetto/ext/base/unix_socket.h" | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <fcntl.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <sys/stat.h> | 
|  | #include <sys/types.h> | 
|  | #include "perfetto/base/compiler.h" | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | // The include order matters on these three Windows header groups. | 
|  | #include <Windows.h> | 
|  |  | 
|  | #include <WS2tcpip.h> | 
|  | #include <WinSock2.h> | 
|  |  | 
|  | #include <afunix.h> | 
|  | #else | 
|  | #include <netdb.h> | 
|  | #include <netinet/in.h> | 
|  | #include <netinet/tcp.h> | 
|  | #include <poll.h> | 
|  | #include <sys/socket.h> | 
|  | #include <sys/un.h> | 
|  | #include <unistd.h> | 
|  | #endif | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | #include <sys/ucred.h> | 
|  | #endif | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <memory> | 
|  |  | 
|  | #include "perfetto/base/build_config.h" | 
|  | #include "perfetto/base/logging.h" | 
|  | #include "perfetto/base/task_runner.h" | 
|  | #include "perfetto/base/time.h" | 
|  | #include "perfetto/ext/base/string_utils.h" | 
|  | #include "perfetto/ext/base/utils.h" | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace base { | 
|  |  | 
|  | // The CMSG_* macros use NULL instead of nullptr. | 
|  | // Note: MSVC doesn't have #pragma GCC diagnostic, hence the if __GNUC__. | 
|  | #if defined(__GNUC__) && !PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | #pragma GCC diagnostic push | 
|  | #pragma GCC diagnostic ignored "-Wzero-as-null-pointer-constant" | 
|  | #endif | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Android takes an int instead of socklen_t for the control buffer size. | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) | 
|  | using CBufLenType = size_t; | 
|  | #else | 
|  | using CBufLenType = socklen_t; | 
|  | #endif | 
|  |  | 
|  | // A wrapper around variable-size sockaddr structs. | 
|  | // This is solving the following problem: when calling connect() or bind(), the | 
|  | // caller needs to take care to allocate the right struct (sockaddr_un for | 
|  | // AF_UNIX, sockaddr_in for AF_INET).   Those structs have different sizes and, | 
|  | // more importantly, are bigger than the base struct sockaddr. | 
|  | struct SockaddrAny { | 
|  | SockaddrAny() : size() {} | 
|  | SockaddrAny(const void* addr, socklen_t sz) | 
|  | : data(new char[static_cast<size_t>(sz)]), size(sz) { | 
|  | memcpy(data.get(), addr, static_cast<size_t>(size)); | 
|  | } | 
|  |  | 
|  | const struct sockaddr* addr() const { | 
|  | return reinterpret_cast<const struct sockaddr*>(data.get()); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<char[]> data; | 
|  | socklen_t size; | 
|  | }; | 
|  |  | 
|  | inline int GetSockFamily(SockFamily family) { | 
|  | switch (family) { | 
|  | case SockFamily::kUnix: | 
|  | return AF_UNIX; | 
|  | case SockFamily::kInet: | 
|  | return AF_INET; | 
|  | case SockFamily::kInet6: | 
|  | return AF_INET6; | 
|  | } | 
|  | PERFETTO_CHECK(false);  // For GCC. | 
|  | } | 
|  |  | 
|  | inline int GetSockType(SockType type) { | 
|  | #ifdef SOCK_CLOEXEC | 
|  | constexpr int kSockCloExec = SOCK_CLOEXEC; | 
|  | #else | 
|  | constexpr int kSockCloExec = 0; | 
|  | #endif | 
|  | switch (type) { | 
|  | case SockType::kStream: | 
|  | return SOCK_STREAM | kSockCloExec; | 
|  | case SockType::kDgram: | 
|  | return SOCK_DGRAM | kSockCloExec; | 
|  | case SockType::kSeqPacket: | 
|  | return SOCK_SEQPACKET | kSockCloExec; | 
|  | } | 
|  | PERFETTO_CHECK(false);  // For GCC. | 
|  | } | 
|  |  | 
|  | SockaddrAny MakeSockAddr(SockFamily family, const std::string& socket_name) { | 
|  | switch (family) { | 
|  | case SockFamily::kUnix: { | 
|  | struct sockaddr_un saddr {}; | 
|  | const size_t name_len = socket_name.size(); | 
|  | if (name_len >= sizeof(saddr.sun_path)) { | 
|  | errno = ENAMETOOLONG; | 
|  | return SockaddrAny(); | 
|  | } | 
|  | memcpy(saddr.sun_path, socket_name.data(), name_len); | 
|  | if (saddr.sun_path[0] == '@') { | 
|  | saddr.sun_path[0] = '\0'; | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | // The MSDN blog claims that abstract (non-filesystem based) AF_UNIX | 
|  | // socket are supported, but that doesn't seem true. | 
|  | PERFETTO_ELOG( | 
|  | "Abstract AF_UNIX sockets are not supported on Windows, see " | 
|  | "https://github.com/microsoft/WSL/issues/4240"); | 
|  | return SockaddrAny{}; | 
|  | #endif | 
|  | } | 
|  | saddr.sun_family = AF_UNIX; | 
|  | auto size = static_cast<socklen_t>( | 
|  | __builtin_offsetof(sockaddr_un, sun_path) + name_len + 1); | 
|  | PERFETTO_CHECK(static_cast<size_t>(size) <= sizeof(saddr)); | 
|  | return SockaddrAny(&saddr, size); | 
|  | } | 
|  | case SockFamily::kInet: { | 
|  | auto parts = SplitString(socket_name, ":"); | 
|  | PERFETTO_CHECK(parts.size() == 2); | 
|  | struct addrinfo* addr_info = nullptr; | 
|  | struct addrinfo hints {}; | 
|  | hints.ai_family = AF_INET; | 
|  | PERFETTO_CHECK(getaddrinfo(parts[0].c_str(), parts[1].c_str(), &hints, | 
|  | &addr_info) == 0); | 
|  | PERFETTO_CHECK(addr_info->ai_family == AF_INET); | 
|  | SockaddrAny res(addr_info->ai_addr, | 
|  | static_cast<socklen_t>(addr_info->ai_addrlen)); | 
|  | freeaddrinfo(addr_info); | 
|  | return res; | 
|  | } | 
|  | case SockFamily::kInet6: { | 
|  | auto parts = SplitString(socket_name, "]"); | 
|  | PERFETTO_CHECK(parts.size() == 2); | 
|  | auto address = SplitString(parts[0], "["); | 
|  | PERFETTO_CHECK(address.size() == 1); | 
|  | auto port = SplitString(parts[1], ":"); | 
|  | PERFETTO_CHECK(port.size() == 1); | 
|  | struct addrinfo* addr_info = nullptr; | 
|  | struct addrinfo hints {}; | 
|  | hints.ai_family = AF_INET6; | 
|  | PERFETTO_CHECK(getaddrinfo(address[0].c_str(), port[0].c_str(), &hints, | 
|  | &addr_info) == 0); | 
|  | PERFETTO_CHECK(addr_info->ai_family == AF_INET6); | 
|  | SockaddrAny res(addr_info->ai_addr, | 
|  | static_cast<socklen_t>(addr_info->ai_addrlen)); | 
|  | freeaddrinfo(addr_info); | 
|  | return res; | 
|  | } | 
|  | } | 
|  | PERFETTO_CHECK(false);  // For GCC. | 
|  | } | 
|  |  | 
|  | ScopedSocketHandle CreateSocketHandle(SockFamily family, SockType type) { | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | static bool init_winsock_once = [] { | 
|  | WSADATA ignored{}; | 
|  | return WSAStartup(MAKEWORD(2, 2), &ignored) == 0; | 
|  | }(); | 
|  | PERFETTO_CHECK(init_winsock_once); | 
|  | #endif | 
|  | return ScopedSocketHandle( | 
|  | socket(GetSockFamily(family), GetSockType(type), 0)); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | int CloseSocket(SocketHandle s) { | 
|  | return ::closesocket(s); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // +-----------------------+ | 
|  | // | UnixSocketRaw methods | | 
|  | // +-----------------------+ | 
|  |  | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | // static | 
|  | void UnixSocketRaw::ShiftMsgHdrPosix(size_t n, struct msghdr* msg) { | 
|  | using LenType = decltype(msg->msg_iovlen);  // Mac and Linux don't agree. | 
|  | for (LenType i = 0; i < msg->msg_iovlen; ++i) { | 
|  | struct iovec* vec = &msg->msg_iov[i]; | 
|  | if (n < vec->iov_len) { | 
|  | // We sent a part of this iovec. | 
|  | vec->iov_base = reinterpret_cast<char*>(vec->iov_base) + n; | 
|  | vec->iov_len -= n; | 
|  | msg->msg_iov = vec; | 
|  | msg->msg_iovlen -= i; | 
|  | return; | 
|  | } | 
|  | // We sent the whole iovec. | 
|  | n -= vec->iov_len; | 
|  | } | 
|  | // We sent all the iovecs. | 
|  | PERFETTO_CHECK(n == 0); | 
|  | msg->msg_iovlen = 0; | 
|  | msg->msg_iov = nullptr; | 
|  | } | 
|  |  | 
|  | // static | 
|  | std::pair<UnixSocketRaw, UnixSocketRaw> UnixSocketRaw::CreatePairPosix( | 
|  | SockFamily family, | 
|  | SockType type) { | 
|  | int fds[2]; | 
|  | if (socketpair(GetSockFamily(family), GetSockType(type), 0, fds) != 0) | 
|  | return std::make_pair(UnixSocketRaw(), UnixSocketRaw()); | 
|  |  | 
|  | return std::make_pair(UnixSocketRaw(ScopedFile(fds[0]), family, type), | 
|  | UnixSocketRaw(ScopedFile(fds[1]), family, type)); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // static | 
|  | UnixSocketRaw UnixSocketRaw::CreateMayFail(SockFamily family, SockType type) { | 
|  | auto fd = CreateSocketHandle(family, type); | 
|  | if (!fd) | 
|  | return UnixSocketRaw(); | 
|  | return UnixSocketRaw(std::move(fd), family, type); | 
|  | } | 
|  |  | 
|  | UnixSocketRaw::UnixSocketRaw() = default; | 
|  |  | 
|  | UnixSocketRaw::UnixSocketRaw(SockFamily family, SockType type) | 
|  | : UnixSocketRaw(CreateSocketHandle(family, type), family, type) {} | 
|  |  | 
|  | UnixSocketRaw::UnixSocketRaw(ScopedSocketHandle fd, | 
|  | SockFamily family, | 
|  | SockType type) | 
|  | : fd_(std::move(fd)), family_(family), type_(type) { | 
|  | PERFETTO_CHECK(fd_); | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | const int no_sigpipe = 1; | 
|  | setsockopt(*fd_, SOL_SOCKET, SO_NOSIGPIPE, &no_sigpipe, sizeof(no_sigpipe)); | 
|  | #endif | 
|  |  | 
|  | if (family == SockFamily::kInet || family == SockFamily::kInet6) { | 
|  | int flag = 1; | 
|  | // The reinterpret_cast<const char*> is needed for Windows, where the 4th | 
|  | // arg is a const char* (on other POSIX system is a const void*). | 
|  | PERFETTO_CHECK(!setsockopt(*fd_, SOL_SOCKET, SO_REUSEADDR, | 
|  | reinterpret_cast<const char*>(&flag), | 
|  | sizeof(flag))); | 
|  | flag = 1; | 
|  | // Disable Nagle's algorithm, optimize for low-latency. | 
|  | // See https://github.com/google/perfetto/issues/70. | 
|  | setsockopt(*fd_, IPPROTO_TCP, TCP_NODELAY, | 
|  | reinterpret_cast<const char*>(&flag), sizeof(flag)); | 
|  | } | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | // We use one event handle for all socket events, to stay consistent to what | 
|  | // we do on UNIX with the base::TaskRunner's poll(). | 
|  | event_handle_.reset(WSACreateEvent()); | 
|  | PERFETTO_CHECK(event_handle_); | 
|  | #else | 
|  | // There is no reason why a socket should outlive the process in case of | 
|  | // exec() by default, this is just working around a broken unix design. | 
|  | int fcntl_res = fcntl(*fd_, F_SETFD, FD_CLOEXEC); | 
|  | PERFETTO_CHECK(fcntl_res == 0); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void UnixSocketRaw::SetBlocking(bool is_blocking) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | unsigned long flag = is_blocking ? 0 : 1;  // FIONBIO has reverse logic. | 
|  | if (is_blocking) { | 
|  | // When switching between non-blocking -> blocking mode, we need to reset | 
|  | // the event handle registration, otherwise the call will fail. | 
|  | PERFETTO_CHECK(WSAEventSelect(*fd_, *event_handle_, 0) == 0); | 
|  | } | 
|  | PERFETTO_CHECK(ioctlsocket(*fd_, static_cast<long>(FIONBIO), &flag) == 0); | 
|  | if (!is_blocking) { | 
|  | PERFETTO_CHECK( | 
|  | WSAEventSelect(*fd_, *event_handle_, | 
|  | FD_ACCEPT | FD_CONNECT | FD_READ | FD_CLOSE) == 0); | 
|  | } | 
|  | #else | 
|  | int flags = fcntl(*fd_, F_GETFL, 0); | 
|  | if (!is_blocking) { | 
|  | flags |= O_NONBLOCK; | 
|  | } else { | 
|  | flags &= ~static_cast<int>(O_NONBLOCK); | 
|  | } | 
|  | int fcntl_res = fcntl(*fd_, F_SETFL, flags); | 
|  | PERFETTO_CHECK(fcntl_res == 0); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void UnixSocketRaw::RetainOnExec() { | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | PERFETTO_DCHECK(fd_); | 
|  | int flags = fcntl(*fd_, F_GETFD, 0); | 
|  | flags &= ~static_cast<int>(FD_CLOEXEC); | 
|  | int fcntl_res = fcntl(*fd_, F_SETFD, flags); | 
|  | PERFETTO_CHECK(fcntl_res == 0); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void UnixSocketRaw::DcheckIsBlocking(bool expected) const { | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | ignore_result(expected); | 
|  | #else | 
|  | PERFETTO_DCHECK(fd_); | 
|  | bool is_blocking = (fcntl(*fd_, F_GETFL, 0) & O_NONBLOCK) == 0; | 
|  | PERFETTO_DCHECK(is_blocking == expected); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | bool UnixSocketRaw::Bind(const std::string& socket_name) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | SockaddrAny addr = MakeSockAddr(family_, socket_name); | 
|  | if (addr.size == 0) | 
|  | return false; | 
|  |  | 
|  | if (bind(*fd_, addr.addr(), addr.size)) { | 
|  | PERFETTO_DPLOG("bind(%s)", socket_name.c_str()); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool UnixSocketRaw::Listen() { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | PERFETTO_DCHECK(type_ == SockType::kStream || type_ == SockType::kSeqPacket); | 
|  | return listen(*fd_, SOMAXCONN) == 0; | 
|  | } | 
|  |  | 
|  | bool UnixSocketRaw::Connect(const std::string& socket_name) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | SockaddrAny addr = MakeSockAddr(family_, socket_name); | 
|  | if (addr.size == 0) | 
|  | return false; | 
|  |  | 
|  | int res = PERFETTO_EINTR(connect(*fd_, addr.addr(), addr.size)); | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | bool continue_async = WSAGetLastError() == WSAEWOULDBLOCK; | 
|  | #else | 
|  | bool continue_async = errno == EINPROGRESS; | 
|  | #endif | 
|  | if (res && !continue_async) | 
|  | return false; | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void UnixSocketRaw::Shutdown() { | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | // Somebody felt very strongly about the naming of this constant. | 
|  | shutdown(*fd_, SD_BOTH); | 
|  | #else | 
|  | shutdown(*fd_, SHUT_RDWR); | 
|  | #endif | 
|  | fd_.reset(); | 
|  | } | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  |  | 
|  | ssize_t UnixSocketRaw::Send(const void* msg, | 
|  | size_t len, | 
|  | const int* /*send_fds*/, | 
|  | size_t num_fds) { | 
|  | PERFETTO_DCHECK(num_fds == 0); | 
|  | return sendto(*fd_, static_cast<const char*>(msg), static_cast<int>(len), 0, | 
|  | nullptr, 0); | 
|  | } | 
|  |  | 
|  | ssize_t UnixSocketRaw::Receive(void* msg, | 
|  | size_t len, | 
|  | ScopedFile* /*fd_vec*/, | 
|  | size_t /*max_files*/) { | 
|  | return recv(*fd_, static_cast<char*>(msg), static_cast<int>(len), 0); | 
|  | } | 
|  |  | 
|  | #else | 
|  | // For the interested reader, Linux kernel dive to verify this is not only a | 
|  | // theoretical possibility: sock_stream_sendmsg, if sock_alloc_send_pskb returns | 
|  | // NULL [1] (which it does when it gets interrupted [2]), returns early with the | 
|  | // amount of bytes already sent. | 
|  | // | 
|  | // [1]: | 
|  | // https://elixir.bootlin.com/linux/v4.18.10/source/net/unix/af_unix.c#L1872 | 
|  | // [2]: https://elixir.bootlin.com/linux/v4.18.10/source/net/core/sock.c#L2101 | 
|  | ssize_t UnixSocketRaw::SendMsgAllPosix(struct msghdr* msg) { | 
|  | // This does not make sense on non-blocking sockets. | 
|  | PERFETTO_DCHECK(fd_); | 
|  |  | 
|  | const bool is_blocking_with_timeout = | 
|  | tx_timeout_ms_ > 0 && ((fcntl(*fd_, F_GETFL, 0) & O_NONBLOCK) == 0); | 
|  | const int64_t start_ms = GetWallTimeMs().count(); | 
|  |  | 
|  | // Waits until some space is available in the tx buffer. | 
|  | // Returns true if some buffer space is available, false if times out. | 
|  | auto poll_or_timeout = [&] { | 
|  | PERFETTO_DCHECK(is_blocking_with_timeout); | 
|  | const int64_t deadline = start_ms + tx_timeout_ms_; | 
|  | const int64_t now_ms = GetWallTimeMs().count(); | 
|  | if (now_ms >= deadline) | 
|  | return false;  // Timed out | 
|  | const int timeout_ms = static_cast<int>(deadline - now_ms); | 
|  | pollfd pfd{*fd_, POLLOUT, 0}; | 
|  | return PERFETTO_EINTR(poll(&pfd, 1, timeout_ms)) > 0; | 
|  | }; | 
|  |  | 
|  | // We implement blocking sends that require a timeout as non-blocking + poll. | 
|  | // This is because SO_SNDTIMEO doesn't work as expected (b/193234818). On linux | 
|  | // we can just pass MSG_DONTWAIT to force the send to be non-blocking. On Mac, | 
|  | // instead we need to flip the O_NONBLOCK flag back and forth. | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | // MSG_NOSIGNAL is not supported on Mac OS X, but in that case the socket is | 
|  | // created with SO_NOSIGPIPE (See InitializeSocket()). | 
|  | int send_flags = 0; | 
|  |  | 
|  | if (is_blocking_with_timeout) | 
|  | SetBlocking(false); | 
|  |  | 
|  | auto reset_nonblock_on_exit = OnScopeExit([&] { | 
|  | if (is_blocking_with_timeout) | 
|  | SetBlocking(true); | 
|  | }); | 
|  | #else | 
|  | int send_flags = MSG_NOSIGNAL | (is_blocking_with_timeout ? MSG_DONTWAIT : 0); | 
|  | #endif | 
|  |  | 
|  | ssize_t total_sent = 0; | 
|  | while (msg->msg_iov) { | 
|  | ssize_t send_res = PERFETTO_EINTR(sendmsg(*fd_, msg, send_flags)); | 
|  | if (send_res == -1 && IsAgain(errno)) { | 
|  | if (is_blocking_with_timeout && poll_or_timeout()) { | 
|  | continue;  // Tx buffer unblocked, repeat the loop. | 
|  | } | 
|  | return total_sent; | 
|  | } else if (send_res <= 0) { | 
|  | return send_res;  // An error occurred. | 
|  | } else { | 
|  | total_sent += send_res; | 
|  | ShiftMsgHdrPosix(static_cast<size_t>(send_res), msg); | 
|  | // Only send the ancillary data with the first sendmsg call. | 
|  | msg->msg_control = nullptr; | 
|  | msg->msg_controllen = 0; | 
|  | } | 
|  | } | 
|  | return total_sent; | 
|  | } | 
|  |  | 
|  | ssize_t UnixSocketRaw::Send(const void* msg, | 
|  | size_t len, | 
|  | const int* send_fds, | 
|  | size_t num_fds) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | msghdr msg_hdr = {}; | 
|  | iovec iov = {const_cast<void*>(msg), len}; | 
|  | msg_hdr.msg_iov = &iov; | 
|  | msg_hdr.msg_iovlen = 1; | 
|  | alignas(cmsghdr) char control_buf[256]; | 
|  |  | 
|  | if (num_fds > 0) { | 
|  | const auto raw_ctl_data_sz = num_fds * sizeof(int); | 
|  | const CBufLenType control_buf_len = | 
|  | static_cast<CBufLenType>(CMSG_SPACE(raw_ctl_data_sz)); | 
|  | PERFETTO_CHECK(control_buf_len <= sizeof(control_buf)); | 
|  | memset(control_buf, 0, sizeof(control_buf)); | 
|  | msg_hdr.msg_control = control_buf; | 
|  | msg_hdr.msg_controllen = control_buf_len;  // used by CMSG_FIRSTHDR | 
|  | struct cmsghdr* cmsg = CMSG_FIRSTHDR(&msg_hdr); | 
|  | cmsg->cmsg_level = SOL_SOCKET; | 
|  | cmsg->cmsg_type = SCM_RIGHTS; | 
|  | cmsg->cmsg_len = static_cast<CBufLenType>(CMSG_LEN(raw_ctl_data_sz)); | 
|  | memcpy(CMSG_DATA(cmsg), send_fds, num_fds * sizeof(int)); | 
|  | // note: if we were to send multiple cmsghdr structures, then | 
|  | // msg_hdr.msg_controllen would need to be adjusted, see "man 3 cmsg". | 
|  | } | 
|  |  | 
|  | return SendMsgAllPosix(&msg_hdr); | 
|  | } | 
|  |  | 
|  | ssize_t UnixSocketRaw::Receive(void* msg, | 
|  | size_t len, | 
|  | ScopedFile* fd_vec, | 
|  | size_t max_files) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | msghdr msg_hdr = {}; | 
|  | iovec iov = {msg, len}; | 
|  | msg_hdr.msg_iov = &iov; | 
|  | msg_hdr.msg_iovlen = 1; | 
|  | alignas(cmsghdr) char control_buf[256]; | 
|  |  | 
|  | if (max_files > 0) { | 
|  | msg_hdr.msg_control = control_buf; | 
|  | msg_hdr.msg_controllen = | 
|  | static_cast<CBufLenType>(CMSG_SPACE(max_files * sizeof(int))); | 
|  | PERFETTO_CHECK(msg_hdr.msg_controllen <= sizeof(control_buf)); | 
|  | } | 
|  | const ssize_t sz = PERFETTO_EINTR(recvmsg(*fd_, &msg_hdr, 0)); | 
|  | if (sz <= 0) { | 
|  | return sz; | 
|  | } | 
|  | PERFETTO_CHECK(static_cast<size_t>(sz) <= len); | 
|  |  | 
|  | int* fds = nullptr; | 
|  | uint32_t fds_len = 0; | 
|  |  | 
|  | if (max_files > 0) { | 
|  | for (cmsghdr* cmsg = CMSG_FIRSTHDR(&msg_hdr); cmsg; | 
|  | cmsg = CMSG_NXTHDR(&msg_hdr, cmsg)) { | 
|  | const size_t payload_len = cmsg->cmsg_len - CMSG_LEN(0); | 
|  | if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS) { | 
|  | PERFETTO_DCHECK(payload_len % sizeof(int) == 0u); | 
|  | PERFETTO_CHECK(fds == nullptr); | 
|  | fds = reinterpret_cast<int*>(CMSG_DATA(cmsg)); | 
|  | fds_len = static_cast<uint32_t>(payload_len / sizeof(int)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (msg_hdr.msg_flags & MSG_TRUNC || msg_hdr.msg_flags & MSG_CTRUNC) { | 
|  | for (size_t i = 0; fds && i < fds_len; ++i) | 
|  | close(fds[i]); | 
|  | PERFETTO_ELOG( | 
|  | "Socket message truncated. This might be due to a SELinux denial on " | 
|  | "fd:use."); | 
|  | errno = EMSGSIZE; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | for (size_t i = 0; fds && i < fds_len; ++i) { | 
|  | if (i < max_files) | 
|  | fd_vec[i].reset(fds[i]); | 
|  | else | 
|  | close(fds[i]); | 
|  | } | 
|  |  | 
|  | return sz; | 
|  | } | 
|  | #endif  // OS_WIN | 
|  |  | 
|  | bool UnixSocketRaw::SetTxTimeout(uint32_t timeout_ms) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | // On Unix-based systems, SO_SNDTIMEO isn't used for Send() because it's | 
|  | // unreliable (b/193234818). Instead we use non-blocking sendmsg() + poll(). | 
|  | // See SendMsgAllPosix(). We still make the setsockopt call because | 
|  | // SO_SNDTIMEO also affects connect(). | 
|  | tx_timeout_ms_ = timeout_ms; | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | DWORD timeout = timeout_ms; | 
|  | ignore_result(tx_timeout_ms_); | 
|  | #else | 
|  | struct timeval timeout {}; | 
|  | uint32_t timeout_sec = timeout_ms / 1000; | 
|  | timeout.tv_sec = static_cast<decltype(timeout.tv_sec)>(timeout_sec); | 
|  | timeout.tv_usec = static_cast<decltype(timeout.tv_usec)>( | 
|  | (timeout_ms - (timeout_sec * 1000)) * 1000); | 
|  | #endif | 
|  | return setsockopt(*fd_, SOL_SOCKET, SO_SNDTIMEO, | 
|  | reinterpret_cast<const char*>(&timeout), | 
|  | sizeof(timeout)) == 0; | 
|  | } | 
|  |  | 
|  | bool UnixSocketRaw::SetRxTimeout(uint32_t timeout_ms) { | 
|  | PERFETTO_DCHECK(fd_); | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | DWORD timeout = timeout_ms; | 
|  | #else | 
|  | struct timeval timeout {}; | 
|  | uint32_t timeout_sec = timeout_ms / 1000; | 
|  | timeout.tv_sec = static_cast<decltype(timeout.tv_sec)>(timeout_sec); | 
|  | timeout.tv_usec = static_cast<decltype(timeout.tv_usec)>( | 
|  | (timeout_ms - (timeout_sec * 1000)) * 1000); | 
|  | #endif | 
|  | return setsockopt(*fd_, SOL_SOCKET, SO_RCVTIMEO, | 
|  | reinterpret_cast<const char*>(&timeout), | 
|  | sizeof(timeout)) == 0; | 
|  | } | 
|  |  | 
|  | #if defined(__GNUC__) && !PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | #pragma GCC diagnostic pop | 
|  | #endif | 
|  |  | 
|  | // +--------------------+ | 
|  | // | UnixSocket methods | | 
|  | // +--------------------+ | 
|  |  | 
|  | // TODO(primiano): Add ThreadChecker to methods of this class. | 
|  |  | 
|  | // static | 
|  | std::unique_ptr<UnixSocket> UnixSocket::Listen(const std::string& socket_name, | 
|  | EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type) { | 
|  | auto sock_raw = UnixSocketRaw::CreateMayFail(sock_family, sock_type); | 
|  | if (!sock_raw || !sock_raw.Bind(socket_name)) | 
|  | return nullptr; | 
|  |  | 
|  | // Forward the call to the Listen() overload below. | 
|  | return Listen(sock_raw.ReleaseFd(), event_listener, task_runner, sock_family, | 
|  | sock_type); | 
|  | } | 
|  |  | 
|  | // static | 
|  | std::unique_ptr<UnixSocket> UnixSocket::Listen(ScopedSocketHandle fd, | 
|  | EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type) { | 
|  | return std::unique_ptr<UnixSocket>(new UnixSocket( | 
|  | event_listener, task_runner, std::move(fd), State::kListening, | 
|  | sock_family, sock_type, SockPeerCredMode::kDefault)); | 
|  | } | 
|  |  | 
|  | // static | 
|  | std::unique_ptr<UnixSocket> UnixSocket::Connect( | 
|  | const std::string& socket_name, | 
|  | EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type, | 
|  | SockPeerCredMode peer_cred_mode) { | 
|  | std::unique_ptr<UnixSocket> sock(new UnixSocket( | 
|  | event_listener, task_runner, sock_family, sock_type, peer_cred_mode)); | 
|  | sock->DoConnect(socket_name); | 
|  | return sock; | 
|  | } | 
|  |  | 
|  | // static | 
|  | std::unique_ptr<UnixSocket> UnixSocket::AdoptConnected( | 
|  | ScopedSocketHandle fd, | 
|  | EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type, | 
|  | SockPeerCredMode peer_cred_mode) { | 
|  | return std::unique_ptr<UnixSocket>(new UnixSocket( | 
|  | event_listener, task_runner, std::move(fd), State::kConnected, | 
|  | sock_family, sock_type, peer_cred_mode)); | 
|  | } | 
|  |  | 
|  | UnixSocket::UnixSocket(EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type, | 
|  | SockPeerCredMode peer_cred_mode) | 
|  | : UnixSocket(event_listener, | 
|  | task_runner, | 
|  | ScopedSocketHandle(), | 
|  | State::kDisconnected, | 
|  | sock_family, | 
|  | sock_type, | 
|  | peer_cred_mode) {} | 
|  |  | 
|  | UnixSocket::UnixSocket(EventListener* event_listener, | 
|  | TaskRunner* task_runner, | 
|  | ScopedSocketHandle adopt_fd, | 
|  | State adopt_state, | 
|  | SockFamily sock_family, | 
|  | SockType sock_type, | 
|  | SockPeerCredMode peer_cred_mode) | 
|  | : peer_cred_mode_(peer_cred_mode), | 
|  | event_listener_(event_listener), | 
|  | task_runner_(task_runner), | 
|  | weak_ptr_factory_(this) { | 
|  | state_ = State::kDisconnected; | 
|  | if (adopt_state == State::kDisconnected) { | 
|  | PERFETTO_DCHECK(!adopt_fd); | 
|  | sock_raw_ = UnixSocketRaw::CreateMayFail(sock_family, sock_type); | 
|  | if (!sock_raw_) | 
|  | return; | 
|  | } else if (adopt_state == State::kConnected) { | 
|  | PERFETTO_DCHECK(adopt_fd); | 
|  | sock_raw_ = UnixSocketRaw(std::move(adopt_fd), sock_family, sock_type); | 
|  | state_ = State::kConnected; | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | if (peer_cred_mode_ == SockPeerCredMode::kReadOnConnect) | 
|  | ReadPeerCredentialsPosix(); | 
|  | #endif | 
|  | } else if (adopt_state == State::kListening) { | 
|  | // We get here from Listen(). | 
|  |  | 
|  | // |adopt_fd| might genuinely be invalid if the bind() failed. | 
|  | if (!adopt_fd) | 
|  | return; | 
|  |  | 
|  | sock_raw_ = UnixSocketRaw(std::move(adopt_fd), sock_family, sock_type); | 
|  | if (!sock_raw_.Listen()) { | 
|  | PERFETTO_DPLOG("listen() failed"); | 
|  | return; | 
|  | } | 
|  | state_ = State::kListening; | 
|  | } else { | 
|  | PERFETTO_FATAL("Unexpected adopt_state");  // Unfeasible. | 
|  | } | 
|  |  | 
|  | PERFETTO_CHECK(sock_raw_); | 
|  |  | 
|  | sock_raw_.SetBlocking(false); | 
|  |  | 
|  | WeakPtr<UnixSocket> weak_ptr = weak_ptr_factory_.GetWeakPtr(); | 
|  |  | 
|  | task_runner_->AddFileDescriptorWatch(sock_raw_.watch_handle(), [weak_ptr] { | 
|  | if (weak_ptr) | 
|  | weak_ptr->OnEvent(); | 
|  | }); | 
|  | } | 
|  |  | 
|  | UnixSocket::~UnixSocket() { | 
|  | // The implicit dtor of |weak_ptr_factory_| will no-op pending callbacks. | 
|  | Shutdown(true); | 
|  | } | 
|  |  | 
|  | UnixSocketRaw UnixSocket::ReleaseSocket() { | 
|  | // This will invalidate any pending calls to OnEvent. | 
|  | state_ = State::kDisconnected; | 
|  | if (sock_raw_) | 
|  | task_runner_->RemoveFileDescriptorWatch(sock_raw_.watch_handle()); | 
|  |  | 
|  | return std::move(sock_raw_); | 
|  | } | 
|  |  | 
|  | // Called only by the Connect() static constructor. | 
|  | void UnixSocket::DoConnect(const std::string& socket_name) { | 
|  | PERFETTO_DCHECK(state_ == State::kDisconnected); | 
|  |  | 
|  | // This is the only thing that can gracefully fail in the ctor. | 
|  | if (!sock_raw_) | 
|  | return NotifyConnectionState(false); | 
|  |  | 
|  | if (!sock_raw_.Connect(socket_name)) | 
|  | return NotifyConnectionState(false); | 
|  |  | 
|  | // At this point either connect() succeeded or started asynchronously | 
|  | // (errno = EINPROGRESS). | 
|  | state_ = State::kConnecting; | 
|  |  | 
|  | // Even if the socket is non-blocking, connecting to a UNIX socket can be | 
|  | // acknowledged straight away rather than returning EINPROGRESS. | 
|  | // The decision here is to deal with the two cases uniformly, at the cost of | 
|  | // delaying the straight-away-connect() case by one task, to avoid depending | 
|  | // on implementation details of UNIX socket on the various OSes. | 
|  | // Posting the OnEvent() below emulates a wakeup of the FD watch. OnEvent(), | 
|  | // which knows how to deal with spurious wakeups, will poll the SO_ERROR and | 
|  | // evolve, if necessary, the state into either kConnected or kDisconnected. | 
|  | WeakPtr<UnixSocket> weak_ptr = weak_ptr_factory_.GetWeakPtr(); | 
|  | task_runner_->PostTask([weak_ptr] { | 
|  | if (weak_ptr) | 
|  | weak_ptr->OnEvent(); | 
|  | }); | 
|  | } | 
|  |  | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | void UnixSocket::ReadPeerCredentialsPosix() { | 
|  | // Peer credentials are supported only on AF_UNIX sockets. | 
|  | if (sock_raw_.family() != SockFamily::kUnix) | 
|  | return; | 
|  | PERFETTO_CHECK(peer_cred_mode_ != SockPeerCredMode::kIgnore); | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \ | 
|  | PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) | 
|  | struct ucred user_cred; | 
|  | socklen_t len = sizeof(user_cred); | 
|  | int fd = sock_raw_.fd(); | 
|  | int res = getsockopt(fd, SOL_SOCKET, SO_PEERCRED, &user_cred, &len); | 
|  | PERFETTO_CHECK(res == 0); | 
|  | peer_uid_ = user_cred.uid; | 
|  | peer_pid_ = user_cred.pid; | 
|  | #elif PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) | 
|  | struct xucred user_cred; | 
|  | socklen_t len = sizeof(user_cred); | 
|  | int res = getsockopt(sock_raw_.fd(), 0, LOCAL_PEERCRED, &user_cred, &len); | 
|  | PERFETTO_CHECK(res == 0 && user_cred.cr_version == XUCRED_VERSION); | 
|  | peer_uid_ = static_cast<uid_t>(user_cred.cr_uid); | 
|  | // There is no pid in the LOCAL_PEERCREDS for MacOS / FreeBSD. | 
|  | #endif | 
|  | } | 
|  | #endif  // !OS_WIN | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | void UnixSocket::OnEvent() { | 
|  | WSANETWORKEVENTS evts{}; | 
|  | PERFETTO_CHECK(WSAEnumNetworkEvents(sock_raw_.fd(), sock_raw_.watch_handle(), | 
|  | &evts) == 0); | 
|  | if (state_ == State::kDisconnected) | 
|  | return;  // Some spurious event, typically queued just before Shutdown(). | 
|  |  | 
|  | if (state_ == State::kConnecting && (evts.lNetworkEvents & FD_CONNECT)) { | 
|  | PERFETTO_DCHECK(sock_raw_); | 
|  | int err = evts.iErrorCode[FD_CONNECT_BIT]; | 
|  | if (err) { | 
|  | PERFETTO_DPLOG("Connection error: %d", err); | 
|  | Shutdown(false); | 
|  | event_listener_->OnConnect(this, false /* connected */); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // kReadOnConnect is not supported on Windows. | 
|  | PERFETTO_DCHECK(peer_cred_mode_ != SockPeerCredMode::kReadOnConnect); | 
|  | state_ = State::kConnected; | 
|  | event_listener_->OnConnect(this, true /* connected */); | 
|  | } | 
|  |  | 
|  | // This is deliberately NOT an else-if. When a client socket connects and | 
|  | // there is already data queued, the following will happen within the same | 
|  | // OnEvent() call: | 
|  | // 1. The block above will transition kConnecting -> kConnected. | 
|  | // 2. This block will cause an OnDataAvailable() call. | 
|  | // Unlike UNIX, where poll() keeps signalling the event until the client | 
|  | // does a recv(), Windows is more picky and stops signalling the event until | 
|  | // the next call to recv() is made. In other words, in Windows we cannot | 
|  | // miss an OnDataAvailable() call or the event pump will stop. | 
|  | if (state_ == State::kConnected) { | 
|  | if (evts.lNetworkEvents & FD_READ) { | 
|  | event_listener_->OnDataAvailable(this); | 
|  | // TODO(primiano): I am very conflicted here. Because of the behavior | 
|  | // described above, if the event listener doesn't do a Recv() call in | 
|  | // the OnDataAvailable() callback, WinSock won't notify the event ever | 
|  | // again. On one side, I don't see any reason why a client should decide | 
|  | // to not do a Recv() in OnDataAvailable. On the other side, the | 
|  | // behavior here diverges from UNIX, where OnDataAvailable() would be | 
|  | // re-posted immediately. In both cases, not doing a Recv() in | 
|  | // OnDataAvailable, leads to something bad (getting stuck on Windows, | 
|  | // getting in a hot loop on Linux), so doesn't feel we should worry too | 
|  | // much about this. If we wanted to keep the behavrior consistent, here | 
|  | // we should do something like: `if (sock_raw_) | 
|  | // sock_raw_.SetBlocking(false)` (Note that the socket might be closed | 
|  | // by the time we come back here, hence the if part). | 
|  | return; | 
|  | } | 
|  | // Could read EOF and disconnect here. | 
|  | if (evts.lNetworkEvents & FD_CLOSE) { | 
|  | Shutdown(true); | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | // New incoming connection. | 
|  | if (state_ == State::kListening && (evts.lNetworkEvents & FD_ACCEPT)) { | 
|  | // There could be more than one incoming connection behind each FD watch | 
|  | // notification. Drain'em all. | 
|  | for (;;) { | 
|  | // Note: right now we don't need the remote endpoint, hence we pass | 
|  | // nullptr to |addr| and |addrlen|. If we ever need to do so, be | 
|  | // extremely careful. Windows' WinSock API will happily write more than | 
|  | // |addrlen| (hence corrupt the stack) if the |addr| argument passed is | 
|  | // not big enough (e.g. passing a struct sockaddr_in to a AF_UNIX | 
|  | // socket, where sizeof(sockaddr_un) is >> sizef(sockaddr_in)). It seems | 
|  | // a Windows / CRT bug in the AF_UNIX implementation. | 
|  | ScopedSocketHandle new_fd(accept(sock_raw_.fd(), nullptr, nullptr)); | 
|  | if (!new_fd) | 
|  | return; | 
|  | std::unique_ptr<UnixSocket> new_sock(new UnixSocket( | 
|  | event_listener_, task_runner_, std::move(new_fd), State::kConnected, | 
|  | sock_raw_.family(), sock_raw_.type(), peer_cred_mode_)); | 
|  | event_listener_->OnNewIncomingConnection(this, std::move(new_sock)); | 
|  | } | 
|  | } | 
|  | } | 
|  | #else | 
|  | void UnixSocket::OnEvent() { | 
|  | if (state_ == State::kDisconnected) | 
|  | return;  // Some spurious event, typically queued just before Shutdown(). | 
|  |  | 
|  | if (state_ == State::kConnected) | 
|  | return event_listener_->OnDataAvailable(this); | 
|  |  | 
|  | if (state_ == State::kConnecting) { | 
|  | PERFETTO_DCHECK(sock_raw_); | 
|  | int sock_err = EINVAL; | 
|  | socklen_t err_len = sizeof(sock_err); | 
|  | int res = | 
|  | getsockopt(sock_raw_.fd(), SOL_SOCKET, SO_ERROR, &sock_err, &err_len); | 
|  |  | 
|  | if (res == 0 && sock_err == EINPROGRESS) | 
|  | return;  // Not connected yet, just a spurious FD watch wakeup. | 
|  | if (res == 0 && sock_err == 0) { | 
|  | if (peer_cred_mode_ == SockPeerCredMode::kReadOnConnect) | 
|  | ReadPeerCredentialsPosix(); | 
|  | state_ = State::kConnected; | 
|  | return event_listener_->OnConnect(this, true /* connected */); | 
|  | } | 
|  | PERFETTO_DLOG("Connection error: %s", strerror(sock_err)); | 
|  | Shutdown(false); | 
|  | return event_listener_->OnConnect(this, false /* connected */); | 
|  | } | 
|  |  | 
|  | // New incoming connection. | 
|  | if (state_ == State::kListening) { | 
|  | // There could be more than one incoming connection behind each FD watch | 
|  | // notification. Drain'em all. | 
|  | for (;;) { | 
|  | ScopedFile new_fd( | 
|  | PERFETTO_EINTR(accept(sock_raw_.fd(), nullptr, nullptr))); | 
|  | if (!new_fd) | 
|  | return; | 
|  | std::unique_ptr<UnixSocket> new_sock(new UnixSocket( | 
|  | event_listener_, task_runner_, std::move(new_fd), State::kConnected, | 
|  | sock_raw_.family(), sock_raw_.type(), peer_cred_mode_)); | 
|  | event_listener_->OnNewIncomingConnection(this, std::move(new_sock)); | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | bool UnixSocket::Send(const void* msg, | 
|  | size_t len, | 
|  | const int* send_fds, | 
|  | size_t num_fds) { | 
|  | if (state_ != State::kConnected) { | 
|  | errno = ENOTCONN; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | sock_raw_.SetBlocking(true); | 
|  | const ssize_t sz = sock_raw_.Send(msg, len, send_fds, num_fds); | 
|  | sock_raw_.SetBlocking(false); | 
|  |  | 
|  | if (sz == static_cast<ssize_t>(len)) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // If we ever decide to support non-blocking sends again, here we should | 
|  | // watch for both EAGAIN and EWOULDBLOCK (see base::IsAgain()). | 
|  |  | 
|  | // If sendmsg() succeeds but the returned size is >= 0 and < |len| it means | 
|  | // that the endpoint disconnected in the middle of the read, and we managed | 
|  | // to send only a portion of the buffer. | 
|  | // If sz < 0, either the other endpoint disconnected (ECONNRESET) or some | 
|  | // other error happened. In both cases we should just give up. | 
|  | PERFETTO_DPLOG("sendmsg() failed"); | 
|  | Shutdown(true); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void UnixSocket::Shutdown(bool notify) { | 
|  | WeakPtr<UnixSocket> weak_ptr = weak_ptr_factory_.GetWeakPtr(); | 
|  | if (notify) { | 
|  | if (state_ == State::kConnected) { | 
|  | task_runner_->PostTask([weak_ptr] { | 
|  | if (weak_ptr) | 
|  | weak_ptr->event_listener_->OnDisconnect(weak_ptr.get()); | 
|  | }); | 
|  | } else if (state_ == State::kConnecting) { | 
|  | task_runner_->PostTask([weak_ptr] { | 
|  | if (weak_ptr) | 
|  | weak_ptr->event_listener_->OnConnect(weak_ptr.get(), false); | 
|  | }); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (sock_raw_) { | 
|  | task_runner_->RemoveFileDescriptorWatch(sock_raw_.watch_handle()); | 
|  | sock_raw_.Shutdown(); | 
|  | } | 
|  | state_ = State::kDisconnected; | 
|  | } | 
|  |  | 
|  | size_t UnixSocket::Receive(void* msg, | 
|  | size_t len, | 
|  | ScopedFile* fd_vec, | 
|  | size_t max_files) { | 
|  | if (state_ != State::kConnected) | 
|  | return 0; | 
|  |  | 
|  | const ssize_t sz = sock_raw_.Receive(msg, len, fd_vec, max_files); | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | bool async_would_block = WSAGetLastError() == WSAEWOULDBLOCK; | 
|  | #else | 
|  | bool async_would_block = IsAgain(errno); | 
|  | #endif | 
|  | if (sz < 0 && async_would_block) | 
|  | return 0; | 
|  |  | 
|  | if (sz <= 0) { | 
|  | Shutdown(true); | 
|  | return 0; | 
|  | } | 
|  | PERFETTO_CHECK(static_cast<size_t>(sz) <= len); | 
|  | return static_cast<size_t>(sz); | 
|  | } | 
|  |  | 
|  | std::string UnixSocket::ReceiveString(size_t max_length) { | 
|  | std::unique_ptr<char[]> buf(new char[max_length + 1]); | 
|  | size_t rsize = Receive(buf.get(), max_length); | 
|  | PERFETTO_CHECK(rsize <= max_length); | 
|  | buf[rsize] = '\0'; | 
|  | return std::string(buf.get()); | 
|  | } | 
|  |  | 
|  | void UnixSocket::NotifyConnectionState(bool success) { | 
|  | if (!success) | 
|  | Shutdown(false); | 
|  |  | 
|  | WeakPtr<UnixSocket> weak_ptr = weak_ptr_factory_.GetWeakPtr(); | 
|  | task_runner_->PostTask([weak_ptr, success] { | 
|  | if (weak_ptr) | 
|  | weak_ptr->event_listener_->OnConnect(weak_ptr.get(), success); | 
|  | }); | 
|  | } | 
|  |  | 
|  | UnixSocket::EventListener::~EventListener() {} | 
|  | void UnixSocket::EventListener::OnNewIncomingConnection( | 
|  | UnixSocket*, | 
|  | std::unique_ptr<UnixSocket>) {} | 
|  | void UnixSocket::EventListener::OnConnect(UnixSocket*, bool) {} | 
|  | void UnixSocket::EventListener::OnDisconnect(UnixSocket*) {} | 
|  | void UnixSocket::EventListener::OnDataAvailable(UnixSocket*) {} | 
|  |  | 
|  | }  // namespace base | 
|  | }  // namespace perfetto |