|
|
|
@ -60,6 +60,7 @@ |
|
|
|
|
#include <grpc/support/string_util.h> |
|
|
|
|
#include <grpc/support/sync.h> |
|
|
|
|
#include <grpc/support/time.h> |
|
|
|
|
#include "src/core/lib/iomgr/error.h" |
|
|
|
|
#include "src/core/lib/iomgr/ev_posix.h" |
|
|
|
|
#include "src/core/lib/iomgr/resolve_address.h" |
|
|
|
|
#include "src/core/lib/iomgr/sockaddr_utils.h" |
|
|
|
@ -128,7 +129,7 @@ grpc_udp_server *grpc_udp_server_create(void) { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void finish_shutdown(grpc_exec_ctx *exec_ctx, grpc_udp_server *s) { |
|
|
|
|
grpc_exec_ctx_enqueue(exec_ctx, s->shutdown_complete, 1, NULL); |
|
|
|
|
grpc_exec_ctx_sched(exec_ctx, s->shutdown_complete, GRPC_ERROR_NONE, NULL); |
|
|
|
|
|
|
|
|
|
gpr_mu_destroy(&s->mu); |
|
|
|
|
gpr_cv_destroy(&s->cv); |
|
|
|
@ -138,7 +139,7 @@ static void finish_shutdown(grpc_exec_ctx *exec_ctx, grpc_udp_server *s) { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void destroyed_port(grpc_exec_ctx *exec_ctx, void *server, |
|
|
|
|
bool success) { |
|
|
|
|
grpc_error *error) { |
|
|
|
|
grpc_udp_server *s = server; |
|
|
|
|
gpr_mu_lock(&s->mu); |
|
|
|
|
s->destroyed_ports++; |
|
|
|
@ -217,14 +218,23 @@ static int prepare_socket(int fd, const struct sockaddr *addr, |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (!grpc_set_socket_nonblocking(fd, 1) || !grpc_set_socket_cloexec(fd, 1)) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to configure socket %d: %s", fd, |
|
|
|
|
strerror(errno)); |
|
|
|
|
if (grpc_set_socket_nonblocking(fd, 1) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to set nonblocking %d: %s", fd, strerror(errno)); |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
if (grpc_set_socket_cloexec(fd, 1) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to set cloexec %d: %s", fd, strerror(errno)); |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (grpc_set_socket_ip_pktinfo_if_possible(fd) && |
|
|
|
|
addr->sa_family == AF_INET6) { |
|
|
|
|
grpc_set_socket_ipv6_recvpktinfo_if_possible(fd); |
|
|
|
|
if (grpc_set_socket_ip_pktinfo_if_possible(fd) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to set ip_pktinfo."); |
|
|
|
|
goto error; |
|
|
|
|
} else if (addr->sa_family == AF_INET6) { |
|
|
|
|
if (grpc_set_socket_ipv6_recvpktinfo_if_possible(fd) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to set ipv6_recvpktinfo."); |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
GPR_ASSERT(addr_len < ~(socklen_t)0); |
|
|
|
@ -241,13 +251,13 @@ static int prepare_socket(int fd, const struct sockaddr *addr, |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (!grpc_set_socket_sndbuf(fd, buffer_size_bytes)) { |
|
|
|
|
if (grpc_set_socket_sndbuf(fd, buffer_size_bytes) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Failed to set send buffer size to %d bytes", |
|
|
|
|
buffer_size_bytes); |
|
|
|
|
goto error; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (!grpc_set_socket_rcvbuf(fd, buffer_size_bytes)) { |
|
|
|
|
if (grpc_set_socket_rcvbuf(fd, buffer_size_bytes) != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_log(GPR_ERROR, "Failed to set receive buffer size to %d bytes", |
|
|
|
|
buffer_size_bytes); |
|
|
|
|
goto error; |
|
|
|
@ -263,10 +273,10 @@ error: |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* event manager callback when reads are ready */ |
|
|
|
|
static void on_read(grpc_exec_ctx *exec_ctx, void *arg, bool success) { |
|
|
|
|
static void on_read(grpc_exec_ctx *exec_ctx, void *arg, grpc_error *error) { |
|
|
|
|
server_port *sp = arg; |
|
|
|
|
|
|
|
|
|
if (!success) { |
|
|
|
|
if (error != GRPC_ERROR_NONE) { |
|
|
|
|
gpr_mu_lock(&sp->server->mu); |
|
|
|
|
if (0 == --sp->server->active_ports) { |
|
|
|
|
gpr_mu_unlock(&sp->server->mu); |
|
|
|
@ -369,7 +379,8 @@ int grpc_udp_server_add_port(grpc_udp_server *s, const void *addr, |
|
|
|
|
/* Try listening on IPv6 first. */ |
|
|
|
|
addr = (struct sockaddr *)&wild6; |
|
|
|
|
addr_len = sizeof(wild6); |
|
|
|
|
fd = grpc_create_dualstack_socket(addr, SOCK_DGRAM, IPPROTO_UDP, &dsmode); |
|
|
|
|
// TODO(rjshade): Test and propagate the returned grpc_error*:
|
|
|
|
|
grpc_create_dualstack_socket(addr, SOCK_DGRAM, IPPROTO_UDP, &dsmode, &fd); |
|
|
|
|
allocated_port1 = |
|
|
|
|
add_socket_to_server(s, fd, addr, addr_len, read_cb, orphan_cb); |
|
|
|
|
if (fd >= 0 && dsmode == GRPC_DSMODE_DUALSTACK) { |
|
|
|
@ -384,7 +395,8 @@ int grpc_udp_server_add_port(grpc_udp_server *s, const void *addr, |
|
|
|
|
addr_len = sizeof(wild4); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
fd = grpc_create_dualstack_socket(addr, SOCK_DGRAM, IPPROTO_UDP, &dsmode); |
|
|
|
|
// TODO(rjshade): Test and propagate the returned grpc_error*:
|
|
|
|
|
grpc_create_dualstack_socket(addr, SOCK_DGRAM, IPPROTO_UDP, &dsmode, &fd); |
|
|
|
|
if (fd < 0) { |
|
|
|
|
gpr_log(GPR_ERROR, "Unable to create socket: %s", strerror(errno)); |
|
|
|
|
} |
|
|
|
|