335 lines
7.8 KiB
C++
335 lines
7.8 KiB
C++
// Copyright 2019 Google LLC
|
|
//
|
|
// 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
|
|
//
|
|
// https://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 "socket.h"
|
|
|
|
#include "rwmutex.h"
|
|
|
|
#if defined(_WIN32)
|
|
#include <winsock2.h>
|
|
#include <ws2tcpip.h>
|
|
#else
|
|
#include <netdb.h>
|
|
#include <netinet/in.h>
|
|
#include <netinet/tcp.h>
|
|
#include <sys/select.h>
|
|
#include <sys/socket.h>
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#if defined(_WIN32)
|
|
#include <atomic>
|
|
namespace {
|
|
std::atomic<int> wsaInitCount = {0};
|
|
} // anonymous namespace
|
|
#else
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
namespace {
|
|
using SOCKET = int;
|
|
} // anonymous namespace
|
|
#endif
|
|
|
|
namespace {
|
|
constexpr SOCKET InvalidSocket = static_cast<SOCKET>(-1);
|
|
void init() {
|
|
#if defined(_WIN32)
|
|
if (wsaInitCount++ == 0) {
|
|
WSADATA winsockData;
|
|
(void)WSAStartup(MAKEWORD(2, 2), &winsockData);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void term() {
|
|
#if defined(_WIN32)
|
|
if (--wsaInitCount == 0) {
|
|
WSACleanup();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
bool setBlocking(SOCKET s, bool blocking) {
|
|
#if defined(_WIN32)
|
|
u_long mode = blocking ? 0 : 1;
|
|
return ioctlsocket(s, FIONBIO, &mode) == NO_ERROR;
|
|
#else
|
|
auto arg = fcntl(s, F_GETFL, nullptr);
|
|
if (arg < 0) {
|
|
return false;
|
|
}
|
|
arg = blocking ? (arg & ~O_NONBLOCK) : (arg | O_NONBLOCK);
|
|
return fcntl(s, F_SETFL, arg) >= 0;
|
|
#endif
|
|
}
|
|
|
|
bool errored(SOCKET s) {
|
|
if (s == InvalidSocket) {
|
|
return true;
|
|
}
|
|
char error = 0;
|
|
socklen_t len = sizeof(error);
|
|
getsockopt(s, SOL_SOCKET, SO_ERROR, &error, &len);
|
|
return error != 0;
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
class dap::Socket::Shared : public dap::ReaderWriter {
|
|
public:
|
|
static std::shared_ptr<Shared> create(const char* address, const char* port) {
|
|
init();
|
|
|
|
addrinfo hints = {};
|
|
hints.ai_family = AF_INET;
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
hints.ai_protocol = IPPROTO_TCP;
|
|
hints.ai_flags = AI_PASSIVE;
|
|
|
|
addrinfo* info = nullptr;
|
|
getaddrinfo(address, port, &hints, &info);
|
|
|
|
if (info) {
|
|
auto socket =
|
|
::socket(info->ai_family, info->ai_socktype, info->ai_protocol);
|
|
auto out = std::make_shared<Shared>(info, socket);
|
|
out->setOptions();
|
|
return out;
|
|
}
|
|
|
|
term();
|
|
return nullptr;
|
|
}
|
|
|
|
Shared(SOCKET socket) : info(nullptr), s(socket) {}
|
|
Shared(addrinfo* info, SOCKET socket) : info(info), s(socket) {}
|
|
|
|
~Shared() {
|
|
if (info) {
|
|
freeaddrinfo(info);
|
|
}
|
|
close();
|
|
term();
|
|
}
|
|
|
|
template <typename FUNCTION>
|
|
void lock(FUNCTION&& f) {
|
|
RLock l(mutex);
|
|
f(s, info);
|
|
}
|
|
|
|
void setOptions() {
|
|
RLock l(mutex);
|
|
if (s == InvalidSocket) {
|
|
return;
|
|
}
|
|
|
|
int enable = 1;
|
|
|
|
#if !defined(_WIN32)
|
|
// Prevent sockets lingering after process termination, causing
|
|
// reconnection issues on the same port.
|
|
setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (char*)&enable, sizeof(enable));
|
|
|
|
struct {
|
|
int l_onoff; /* linger active */
|
|
int l_linger; /* how many seconds to linger for */
|
|
} linger = {false, 0};
|
|
setsockopt(s, SOL_SOCKET, SO_LINGER, (char*)&linger, sizeof(linger));
|
|
#endif // !defined(_WIN32)
|
|
|
|
// Enable TCP_NODELAY.
|
|
// DAP usually consists of small packet requests, with small packet
|
|
// responses. When there are many frequent, blocking requests made,
|
|
// Nagle's algorithm can dramatically limit the request->response rates.
|
|
setsockopt(s, IPPROTO_TCP, TCP_NODELAY, (char*)&enable, sizeof(enable));
|
|
}
|
|
|
|
// dap::ReaderWriter compliance
|
|
bool isOpen() {
|
|
{
|
|
RLock l(mutex);
|
|
if ((s != InvalidSocket) && !errored(s)) {
|
|
return true;
|
|
}
|
|
}
|
|
WLock lock(mutex);
|
|
s = InvalidSocket;
|
|
return false;
|
|
}
|
|
|
|
void close() {
|
|
{
|
|
RLock l(mutex);
|
|
if (s != InvalidSocket) {
|
|
#if defined(_WIN32)
|
|
closesocket(s);
|
|
#elif __APPLE__
|
|
// ::shutdown() *should* be sufficient to unblock ::accept(), but
|
|
// apparently on macos it can return ENOTCONN and ::accept() continues
|
|
// to block indefinitely.
|
|
// Note: There is a race here. Calling ::close() frees the socket ID,
|
|
// which may be reused before `s` is assigned InvalidSocket.
|
|
::shutdown(s, SHUT_RDWR);
|
|
::close(s);
|
|
#else
|
|
// ::shutdown() to unblock ::accept(). We'll actually close the socket
|
|
// under lock below.
|
|
::shutdown(s, SHUT_RDWR);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
WLock l(mutex);
|
|
if (s != InvalidSocket) {
|
|
#if !defined(_WIN32) && !defined(__APPLE__)
|
|
::close(s);
|
|
#endif
|
|
s = InvalidSocket;
|
|
}
|
|
}
|
|
|
|
size_t read(void* buffer, size_t bytes) {
|
|
RLock lock(mutex);
|
|
if (s == InvalidSocket) {
|
|
return 0;
|
|
}
|
|
auto len =
|
|
recv(s, reinterpret_cast<char*>(buffer), static_cast<int>(bytes), 0);
|
|
return (len < 0) ? 0 : len;
|
|
}
|
|
|
|
bool write(const void* buffer, size_t bytes) {
|
|
RLock lock(mutex);
|
|
if (s == InvalidSocket) {
|
|
return false;
|
|
}
|
|
if (bytes == 0) {
|
|
return true;
|
|
}
|
|
return ::send(s, reinterpret_cast<const char*>(buffer),
|
|
static_cast<int>(bytes), 0) > 0;
|
|
}
|
|
|
|
private:
|
|
addrinfo* const info;
|
|
SOCKET s = InvalidSocket;
|
|
RWMutex mutex;
|
|
};
|
|
|
|
namespace dap {
|
|
|
|
Socket::Socket(const char* address, const char* port)
|
|
: shared(Shared::create(address, port)) {
|
|
if (shared) {
|
|
shared->lock([&](SOCKET socket, const addrinfo* info) {
|
|
if (bind(socket, info->ai_addr, (int)info->ai_addrlen) != 0) {
|
|
shared.reset();
|
|
return;
|
|
}
|
|
|
|
if (listen(socket, 0) != 0) {
|
|
shared.reset();
|
|
return;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<ReaderWriter> Socket::accept() const {
|
|
std::shared_ptr<Shared> out;
|
|
if (shared) {
|
|
shared->lock([&](SOCKET socket, const addrinfo*) {
|
|
if (socket != InvalidSocket && !errored(socket)) {
|
|
init();
|
|
auto accepted = ::accept(socket, 0, 0);
|
|
if (accepted != InvalidSocket) {
|
|
out = std::make_shared<Shared>(accepted);
|
|
out->setOptions();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
return out;
|
|
}
|
|
|
|
bool Socket::isOpen() const {
|
|
if (shared) {
|
|
return shared->isOpen();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void Socket::close() const {
|
|
if (shared) {
|
|
shared->close();
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<ReaderWriter> Socket::connect(const char* address,
|
|
const char* port,
|
|
uint32_t timeoutMillis) {
|
|
auto shared = Shared::create(address, port);
|
|
if (!shared) {
|
|
return nullptr;
|
|
}
|
|
|
|
std::shared_ptr<ReaderWriter> out;
|
|
shared->lock([&](SOCKET socket, const addrinfo* info) {
|
|
if (socket == InvalidSocket) {
|
|
return;
|
|
}
|
|
|
|
if (timeoutMillis == 0) {
|
|
if (::connect(socket, info->ai_addr, (int)info->ai_addrlen) == 0) {
|
|
out = shared;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (!setBlocking(socket, false)) {
|
|
return;
|
|
}
|
|
|
|
auto res = ::connect(socket, info->ai_addr, (int)info->ai_addrlen);
|
|
if (res == 0) {
|
|
if (setBlocking(socket, true)) {
|
|
out = shared;
|
|
}
|
|
} else {
|
|
const auto microseconds = timeoutMillis * 1000;
|
|
|
|
fd_set fdset;
|
|
FD_ZERO(&fdset);
|
|
FD_SET(socket, &fdset);
|
|
|
|
timeval tv;
|
|
tv.tv_sec = microseconds / 1000000;
|
|
tv.tv_usec = microseconds - static_cast<uint32_t>(tv.tv_sec * 1000000);
|
|
res = select(static_cast<int>(socket + 1), nullptr, &fdset, nullptr, &tv);
|
|
if (res > 0 && !errored(socket) && setBlocking(socket, true)) {
|
|
out = shared;
|
|
}
|
|
}
|
|
});
|
|
|
|
if (!out) {
|
|
return nullptr;
|
|
}
|
|
|
|
return out->isOpen() ? out : nullptr;
|
|
}
|
|
|
|
} // namespace dap
|