Merge pull request #6347 from ctiller/catastrophic_monkey_dance_241

Fix a case whereby we leak a winsocket if we fail to connect
pull/6227/merge
Nicolas Noble 9 years ago
commit 5ff9920248
  1. 29
      src/core/lib/iomgr/tcp_client_windows.c

@ -63,39 +63,45 @@ typedef struct {
grpc_endpoint **endpoint; grpc_endpoint **endpoint;
} async_connect; } async_connect;
static void async_connect_unlock_and_cleanup(async_connect *ac) { static void async_connect_unlock_and_cleanup(async_connect *ac,
grpc_winsocket *socket) {
int done = (--ac->refs == 0); int done = (--ac->refs == 0);
gpr_mu_unlock(&ac->mu); gpr_mu_unlock(&ac->mu);
if (done) { if (done) {
if (ac->socket != NULL) grpc_winsocket_destroy(ac->socket);
gpr_mu_destroy(&ac->mu); gpr_mu_destroy(&ac->mu);
gpr_free(ac->addr_name); gpr_free(ac->addr_name);
gpr_free(ac); gpr_free(ac);
} }
if (socket != NULL) grpc_winsocket_destroy(socket);
} }
static void on_alarm(grpc_exec_ctx *exec_ctx, void *acp, bool occured) { static void on_alarm(grpc_exec_ctx *exec_ctx, void *acp, bool occured) {
async_connect *ac = acp; async_connect *ac = acp;
gpr_mu_lock(&ac->mu); gpr_mu_lock(&ac->mu);
/* If the alarm didn't occur, it got cancelled. */ if (ac->socket != NULL) {
if (ac->socket != NULL && occured) {
grpc_winsocket_shutdown(ac->socket); grpc_winsocket_shutdown(ac->socket);
} }
async_connect_unlock_and_cleanup(ac); async_connect_unlock_and_cleanup(ac, ac->socket);
} }
static void on_connect(grpc_exec_ctx *exec_ctx, void *acp, bool from_iocp) { static void on_connect(grpc_exec_ctx *exec_ctx, void *acp, bool from_iocp) {
async_connect *ac = acp; async_connect *ac = acp;
SOCKET sock = ac->socket->socket; SOCKET sock = ac->socket->socket;
grpc_endpoint **ep = ac->endpoint; grpc_endpoint **ep = ac->endpoint;
GPR_ASSERT(*ep == NULL);
grpc_winsocket_callback_info *info = &ac->socket->write_info; grpc_winsocket_callback_info *info = &ac->socket->write_info;
grpc_closure *on_done = ac->on_done; grpc_closure *on_done = ac->on_done;
gpr_mu_lock(&ac->mu);
grpc_winsocket *socket = ac->socket;
ac->socket = NULL;
gpr_mu_unlock(&ac->mu);
grpc_timer_cancel(exec_ctx, &ac->alarm); grpc_timer_cancel(exec_ctx, &ac->alarm);
gpr_mu_lock(&ac->mu); gpr_mu_lock(&ac->mu);
if (from_iocp) { if (from_iocp && socket != NULL) {
DWORD transfered_bytes = 0; DWORD transfered_bytes = 0;
DWORD flags; DWORD flags;
BOOL wsa_success = WSAGetOverlappedResult(sock, &info->overlapped, BOOL wsa_success = WSAGetOverlappedResult(sock, &info->overlapped,
@ -107,12 +113,12 @@ static void on_connect(grpc_exec_ctx *exec_ctx, void *acp, bool from_iocp) {
ac->addr_name, utf8_message); ac->addr_name, utf8_message);
gpr_free(utf8_message); gpr_free(utf8_message);
} else { } else {
*ep = grpc_tcp_create(ac->socket, ac->addr_name); *ep = grpc_tcp_create(socket, ac->addr_name);
ac->socket = NULL; socket = NULL;
} }
} }
async_connect_unlock_and_cleanup(ac); async_connect_unlock_and_cleanup(ac, socket);
/* If the connection was aborted, the callback was already called when /* If the connection was aborted, the callback was already called when
the deadline was met. */ the deadline was met. */
on_done->cb(exec_ctx, on_done->cb_arg, *ep != NULL); on_done->cb(exec_ctx, on_done->cb_arg, *ep != NULL);
@ -138,6 +144,7 @@ void grpc_tcp_client_connect(grpc_exec_ctx *exec_ctx, grpc_closure *on_done,
const char *message = NULL; const char *message = NULL;
char *utf8_message; char *utf8_message;
grpc_winsocket_callback_info *info; grpc_winsocket_callback_info *info;
int last_error;
*endpoint = NULL; *endpoint = NULL;
@ -208,8 +215,10 @@ void grpc_tcp_client_connect(grpc_exec_ctx *exec_ctx, grpc_closure *on_done,
return; return;
failure: failure:
utf8_message = gpr_format_message(WSAGetLastError()); last_error = WSAGetLastError();
utf8_message = gpr_format_message(last_error);
gpr_log(GPR_ERROR, message, utf8_message); gpr_log(GPR_ERROR, message, utf8_message);
gpr_log(GPR_ERROR, "last error = %d", last_error);
gpr_free(utf8_message); gpr_free(utf8_message);
if (socket != NULL) { if (socket != NULL) {
grpc_winsocket_destroy(socket); grpc_winsocket_destroy(socket);

Loading…
Cancel
Save