//===-- DomainSocket.cpp --------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "lldb/Host/posix/DomainSocket.h" #include "lldb/Utility/LLDBLog.h" #include "llvm/Support/Errno.h" #include "llvm/Support/FileSystem.h" #include #include #include #include using namespace lldb; using namespace lldb_private; static const int kDomain = AF_UNIX; static const int kType = SOCK_STREAM; static bool SetSockAddr(llvm::StringRef name, const size_t name_offset, sockaddr_un *saddr_un, socklen_t &saddr_un_len) { if (name.size() + name_offset > sizeof(saddr_un->sun_path)) return false; memset(saddr_un, 0, sizeof(*saddr_un)); saddr_un->sun_family = kDomain; memcpy(saddr_un->sun_path + name_offset, name.data(), name.size()); // For domain sockets we can use SUN_LEN in order to calculate size of // sockaddr_un, but for abstract sockets we have to calculate size manually // because of leading null symbol. if (name_offset == 0) saddr_un_len = SUN_LEN(saddr_un); else saddr_un_len = offsetof(struct sockaddr_un, sun_path) + name_offset + name.size(); #if defined(__APPLE__) || defined(__FreeBSD__) || defined(__NetBSD__) || \ defined(__OpenBSD__) saddr_un->sun_len = saddr_un_len; #endif return true; } DomainSocket::DomainSocket(bool should_close) : DomainSocket(kInvalidSocketValue, should_close) {} DomainSocket::DomainSocket(NativeSocket socket, bool should_close) : Socket(ProtocolUnixDomain, should_close) { m_socket = socket; } DomainSocket::DomainSocket(SocketProtocol protocol) : Socket(protocol, /*should_close=*/true) {} DomainSocket::DomainSocket(NativeSocket socket, const DomainSocket &listen_socket) : Socket(ProtocolUnixDomain, listen_socket.m_should_close_fd) { m_socket = socket; } Status DomainSocket::Connect(llvm::StringRef name) { sockaddr_un saddr_un; socklen_t saddr_un_len; if (!SetSockAddr(name, GetNameOffset(), &saddr_un, saddr_un_len)) return Status::FromErrorString("Failed to set socket address"); Status error; m_socket = CreateSocket(kDomain, kType, 0, error); if (error.Fail()) return error; if (llvm::sys::RetryAfterSignal(-1, ::connect, GetNativeSocket(), (struct sockaddr *)&saddr_un, saddr_un_len) < 0) SetLastError(error); return error; } Status DomainSocket::Listen(llvm::StringRef name, int backlog) { sockaddr_un saddr_un; socklen_t saddr_un_len; if (!SetSockAddr(name, GetNameOffset(), &saddr_un, saddr_un_len)) return Status::FromErrorString("Failed to set socket address"); DeleteSocketFile(name); Status error; m_socket = CreateSocket(kDomain, kType, 0, error); if (error.Fail()) return error; if (::bind(GetNativeSocket(), (struct sockaddr *)&saddr_un, saddr_un_len) == 0) if (::listen(GetNativeSocket(), backlog) == 0) return error; SetLastError(error); return error; } llvm::Expected> DomainSocket::Accept( MainLoopBase &loop, std::function socket)> sock_cb) { // TODO: Refactor MainLoop to avoid the shared_ptr requirement. auto io_sp = std::make_shared(GetNativeSocket(), false); auto cb = [this, sock_cb](MainLoopBase &loop) { Log *log = GetLog(LLDBLog::Host); Status error; auto conn_fd = AcceptSocket(GetNativeSocket(), nullptr, nullptr, error); if (error.Fail()) { LLDB_LOG(log, "AcceptSocket({0}): {1}", GetNativeSocket(), error); return; } std::unique_ptr sock_up(new DomainSocket(conn_fd, *this)); sock_cb(std::move(sock_up)); }; Status error; std::vector handles; handles.emplace_back(loop.RegisterReadObject(io_sp, cb, error)); if (error.Fail()) return error.ToError(); return handles; } size_t DomainSocket::GetNameOffset() const { return 0; } void DomainSocket::DeleteSocketFile(llvm::StringRef name) { llvm::sys::fs::remove(name); } std::string DomainSocket::GetSocketName() const { if (m_socket == kInvalidSocketValue) return ""; struct sockaddr_un saddr_un; saddr_un.sun_family = AF_UNIX; socklen_t sock_addr_len = sizeof(struct sockaddr_un); if (::getpeername(m_socket, (struct sockaddr *)&saddr_un, &sock_addr_len) != 0) return ""; if (sock_addr_len <= offsetof(struct sockaddr_un, sun_path)) return ""; // Unnamed domain socket llvm::StringRef name(saddr_un.sun_path + GetNameOffset(), sock_addr_len - offsetof(struct sockaddr_un, sun_path) - GetNameOffset()); name = name.rtrim('\0'); return name.str(); } std::string DomainSocket::GetRemoteConnectionURI() const { std::string name = GetSocketName(); if (name.empty()) return name; return llvm::formatv( "{0}://{1}", GetNameOffset() == 0 ? "unix-connect" : "unix-abstract-connect", name); } std::vector DomainSocket::GetListeningConnectionURI() const { if (m_socket == kInvalidSocketValue) return {}; struct sockaddr_un addr; memset(&addr, 0, sizeof(struct sockaddr_un)); addr.sun_family = AF_UNIX; socklen_t addr_len = sizeof(struct sockaddr_un); if (::getsockname(m_socket, (struct sockaddr *)&addr, &addr_len) != 0) return {}; return {llvm::formatv("unix-connect://{0}", addr.sun_path)}; }