Refactor setting socket options (#2053)
Add detail::set_socket_opt() and detail::set_socket_opt_time() to avoid repetition of platform-specific code.
This commit is contained in:
parent
03cf43ebaa
commit
233f0fb1b8
147
httplib.h
147
httplib.h
@ -193,6 +193,7 @@ using ssize_t = long;
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
using socket_t = SOCKET;
|
using socket_t = SOCKET;
|
||||||
|
using socklen_t = int;
|
||||||
#ifdef CPPHTTPLIB_USE_POLL
|
#ifdef CPPHTTPLIB_USE_POLL
|
||||||
#define poll(fds, nfds, timeout) WSAPoll(fds, nfds, timeout)
|
#define poll(fds, nfds, timeout) WSAPoll(fds, nfds, timeout)
|
||||||
#endif
|
#endif
|
||||||
@ -848,6 +849,16 @@ using Logger = std::function<void(const Request &, const Response &)>;
|
|||||||
|
|
||||||
using SocketOptions = std::function<void(socket_t sock)>;
|
using SocketOptions = std::function<void(socket_t sock)>;
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
bool set_socket_opt_impl(socket_t sock, int level, int optname,
|
||||||
|
const void *optval, socklen_t optlen);
|
||||||
|
bool set_socket_opt(socket_t sock, int level, int optname, int opt);
|
||||||
|
bool set_socket_opt_time(socket_t sock, int level, int optname, time_t sec,
|
||||||
|
time_t usec);
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
void default_socket_options(socket_t sock);
|
void default_socket_options(socket_t sock);
|
||||||
|
|
||||||
const char *status_message(int status);
|
const char *status_message(int status);
|
||||||
@ -2075,20 +2086,45 @@ inline uint64_t Response::get_header_value_u64(const std::string &key,
|
|||||||
return detail::get_header_value_u64(headers, key, def, id);
|
return detail::get_header_value_u64(headers, key, def, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void default_socket_options(socket_t sock) {
|
namespace detail {
|
||||||
int opt = 1;
|
|
||||||
|
inline bool set_socket_opt_impl(socket_t sock, int level, int optname,
|
||||||
|
const void *optval, socklen_t optlen) {
|
||||||
|
return setsockopt(sock, level, optname,
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
|
reinterpret_cast<const char *>(optval),
|
||||||
reinterpret_cast<const char *>(&opt), sizeof(opt));
|
|
||||||
#else
|
#else
|
||||||
|
optval,
|
||||||
|
#endif
|
||||||
|
optlen) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool set_socket_opt(socket_t sock, int level, int optname, int optval) {
|
||||||
|
return set_socket_opt_impl(sock, level, optname, &optval, sizeof(optval));
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool set_socket_opt_time(socket_t sock, int level, int optname,
|
||||||
|
time_t sec, time_t usec) {
|
||||||
|
#ifdef _WIN32
|
||||||
|
auto timeout = static_cast<uint32_t>(sec * 1000 + usec / 1000);
|
||||||
|
#else
|
||||||
|
timeval timeout;
|
||||||
|
timeout.tv_sec = static_cast<long>(sec);
|
||||||
|
timeout.tv_usec = static_cast<decltype(timeout.tv_usec)>(usec);
|
||||||
|
#endif
|
||||||
|
return set_socket_opt_impl(sock, level, optname, &timeout, sizeof(timeout));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
|
inline void default_socket_options(socket_t sock) {
|
||||||
|
detail::set_socket_opt(sock, SOL_SOCKET,
|
||||||
#ifdef SO_REUSEPORT
|
#ifdef SO_REUSEPORT
|
||||||
setsockopt(sock, SOL_SOCKET, SO_REUSEPORT,
|
SO_REUSEPORT,
|
||||||
reinterpret_cast<const void *>(&opt), sizeof(opt));
|
|
||||||
#else
|
#else
|
||||||
setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
|
SO_REUSEADDR,
|
||||||
reinterpret_cast<const void *>(&opt), sizeof(opt));
|
|
||||||
#endif
|
|
||||||
#endif
|
#endif
|
||||||
|
1);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline const char *status_message(int status) {
|
inline const char *status_message(int status) {
|
||||||
@ -3605,26 +3641,10 @@ socket_t create_socket(const std::string &host, const std::string &ip, int port,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (tcp_nodelay) {
|
if (tcp_nodelay) { set_socket_opt(sock, IPPROTO_TCP, TCP_NODELAY, 1); }
|
||||||
auto opt = 1;
|
|
||||||
#ifdef _WIN32
|
|
||||||
setsockopt(sock, IPPROTO_TCP, TCP_NODELAY,
|
|
||||||
reinterpret_cast<const char *>(&opt), sizeof(opt));
|
|
||||||
#else
|
|
||||||
setsockopt(sock, IPPROTO_TCP, TCP_NODELAY,
|
|
||||||
reinterpret_cast<const void *>(&opt), sizeof(opt));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rp->ai_family == AF_INET6) {
|
if (rp->ai_family == AF_INET6) {
|
||||||
auto opt = ipv6_v6only ? 1 : 0;
|
set_socket_opt(sock, IPPROTO_IPV6, IPV6_V6ONLY, ipv6_v6only ? 1 : 0);
|
||||||
#ifdef _WIN32
|
|
||||||
setsockopt(sock, IPPROTO_IPV6, IPV6_V6ONLY,
|
|
||||||
reinterpret_cast<const char *>(&opt), sizeof(opt));
|
|
||||||
#else
|
|
||||||
setsockopt(sock, IPPROTO_IPV6, IPV6_V6ONLY,
|
|
||||||
reinterpret_cast<const void *>(&opt), sizeof(opt));
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (socket_options) { socket_options(sock); }
|
if (socket_options) { socket_options(sock); }
|
||||||
@ -3767,36 +3787,10 @@ inline socket_t create_client_socket(
|
|||||||
}
|
}
|
||||||
|
|
||||||
set_nonblocking(sock2, false);
|
set_nonblocking(sock2, false);
|
||||||
|
set_socket_opt_time(sock2, SOL_SOCKET, SO_RCVTIMEO, read_timeout_sec,
|
||||||
{
|
read_timeout_usec);
|
||||||
#ifdef _WIN32
|
set_socket_opt_time(sock2, SOL_SOCKET, SO_SNDTIMEO, write_timeout_sec,
|
||||||
auto timeout = static_cast<uint32_t>(read_timeout_sec * 1000 +
|
write_timeout_usec);
|
||||||
read_timeout_usec / 1000);
|
|
||||||
setsockopt(sock2, SOL_SOCKET, SO_RCVTIMEO,
|
|
||||||
reinterpret_cast<const char *>(&timeout), sizeof(timeout));
|
|
||||||
#else
|
|
||||||
timeval tv;
|
|
||||||
tv.tv_sec = static_cast<long>(read_timeout_sec);
|
|
||||||
tv.tv_usec = static_cast<decltype(tv.tv_usec)>(read_timeout_usec);
|
|
||||||
setsockopt(sock2, SOL_SOCKET, SO_RCVTIMEO,
|
|
||||||
reinterpret_cast<const void *>(&tv), sizeof(tv));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
auto timeout = static_cast<uint32_t>(write_timeout_sec * 1000 +
|
|
||||||
write_timeout_usec / 1000);
|
|
||||||
setsockopt(sock2, SOL_SOCKET, SO_SNDTIMEO,
|
|
||||||
reinterpret_cast<const char *>(&timeout), sizeof(timeout));
|
|
||||||
#else
|
|
||||||
timeval tv;
|
|
||||||
tv.tv_sec = static_cast<long>(write_timeout_sec);
|
|
||||||
tv.tv_usec = static_cast<decltype(tv.tv_usec)>(write_timeout_usec);
|
|
||||||
setsockopt(sock2, SOL_SOCKET, SO_SNDTIMEO,
|
|
||||||
reinterpret_cast<const void *>(&tv), sizeof(tv));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
error = Error::Success;
|
error = Error::Success;
|
||||||
return true;
|
return true;
|
||||||
@ -6946,35 +6940,10 @@ inline bool Server::listen_internal() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
detail::set_socket_opt_time(sock, SOL_SOCKET, SO_RCVTIMEO,
|
||||||
#ifdef _WIN32
|
read_timeout_sec_, read_timeout_usec_);
|
||||||
auto timeout = static_cast<uint32_t>(read_timeout_sec_ * 1000 +
|
detail::set_socket_opt_time(sock, SOL_SOCKET, SO_SNDTIMEO,
|
||||||
read_timeout_usec_ / 1000);
|
write_timeout_sec_, write_timeout_usec_);
|
||||||
setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO,
|
|
||||||
reinterpret_cast<const char *>(&timeout), sizeof(timeout));
|
|
||||||
#else
|
|
||||||
timeval tv;
|
|
||||||
tv.tv_sec = static_cast<long>(read_timeout_sec_);
|
|
||||||
tv.tv_usec = static_cast<decltype(tv.tv_usec)>(read_timeout_usec_);
|
|
||||||
setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO,
|
|
||||||
reinterpret_cast<const void *>(&tv), sizeof(tv));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
auto timeout = static_cast<uint32_t>(write_timeout_sec_ * 1000 +
|
|
||||||
write_timeout_usec_ / 1000);
|
|
||||||
setsockopt(sock, SOL_SOCKET, SO_SNDTIMEO,
|
|
||||||
reinterpret_cast<const char *>(&timeout), sizeof(timeout));
|
|
||||||
#else
|
|
||||||
timeval tv;
|
|
||||||
tv.tv_sec = static_cast<long>(write_timeout_sec_);
|
|
||||||
tv.tv_usec = static_cast<decltype(tv.tv_usec)>(write_timeout_usec_);
|
|
||||||
setsockopt(sock, SOL_SOCKET, SO_SNDTIMEO,
|
|
||||||
reinterpret_cast<const void *>(&tv), sizeof(tv));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!task_queue->enqueue(
|
if (!task_queue->enqueue(
|
||||||
[this, sock]() { process_and_close_socket(sock); })) {
|
[this, sock]() { process_and_close_socket(sock); })) {
|
||||||
@ -9097,11 +9066,7 @@ inline void ssl_delete(std::mutex &ctx_mutex, SSL *ssl, socket_t sock,
|
|||||||
(void)(sock);
|
(void)(sock);
|
||||||
SSL_shutdown(ssl);
|
SSL_shutdown(ssl);
|
||||||
#else
|
#else
|
||||||
timeval tv;
|
detail::set_socket_opt_time(sock, SOL_SOCKET, SO_RCVTIMEO, 1, 0);
|
||||||
tv.tv_sec = 1;
|
|
||||||
tv.tv_usec = 0;
|
|
||||||
setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO,
|
|
||||||
reinterpret_cast<const void *>(&tv), sizeof(tv));
|
|
||||||
|
|
||||||
auto ret = SSL_shutdown(ssl);
|
auto ret = SSL_shutdown(ssl);
|
||||||
while (ret == 0) {
|
while (ret == 0) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user