diff options
-rw-r--r-- | source4/lib/socket/connect.c | 140 |
1 files changed, 72 insertions, 68 deletions
diff --git a/source4/lib/socket/connect.c b/source4/lib/socket/connect.c index 244fdcbb06..1a29667d1c 100644 --- a/source4/lib/socket/connect.c +++ b/source4/lib/socket/connect.c @@ -37,7 +37,7 @@ struct connect_state { const char *server_address; int server_port; uint32_t flags; - struct fd_event *connect_ev; + struct fd_event *fde; }; static void socket_connect_handler(struct event_context *ev, @@ -46,6 +46,44 @@ static void socket_connect_handler(struct event_context *ev, static void socket_connect_recv_addr(struct composite_context *ctx); static void socket_connect_recv_conn(struct composite_context *ctx); +/* + call the real socket_connect() call, and setup event handler +*/ +static void socket_send_connect(struct connect_state *state) +{ + struct composite_context *ctx; + + ctx = talloc_zero(state, struct composite_context); + if (composite_nomem(ctx, state->ctx)) return; + ctx->state = COMPOSITE_STATE_IN_PROGRESS; + ctx->event_ctx = state->ctx->event_ctx; + ctx->async.fn = socket_connect_recv_conn; + ctx->async.private_data = state; + + state->ctx->status = socket_connect(state->sock, + state->my_address, + state->my_port, + state->server_address, + state->server_port, + state->flags); + if (NT_STATUS_IS_ERR(state->ctx->status) && + !NT_STATUS_EQUAL(state->ctx->status, + NT_STATUS_MORE_PROCESSING_REQUIRED)) { + composite_error(state->ctx, state->ctx->status); + return; + } + + state->fde = event_add_fd(ctx->event_ctx, state, + socket_get_fd(state->sock), + EVENT_FD_WRITE, + socket_connect_handler, ctx); + composite_nomem(state->fde, state->ctx); +} + + +/* + send a socket connect, potentially doing some name resolution first +*/ struct composite_context *socket_connect_send(struct socket_context *sock, const char *my_address, int my_port, @@ -57,24 +95,29 @@ struct composite_context *socket_connect_send(struct socket_context *sock, struct composite_context *result, *ctx; struct connect_state *state; - result = talloc_zero(NULL, struct composite_context); - if (result == NULL) goto failed; + result = talloc_zero(sock, struct composite_context); + if (result == NULL) return NULL; result->state = COMPOSITE_STATE_IN_PROGRESS; result->async.fn = NULL; result->event_ctx = event_ctx; - state = talloc(result, struct connect_state); - if (state == NULL) goto failed; + state = talloc_zero(result, struct connect_state); + if (composite_nomem(state, result)) goto failed; state->ctx = result; result->private_data = state; state->sock = talloc_reference(state, sock); - if (state->sock == NULL) goto failed; - state->my_address = talloc_strdup(state, my_address); - if (state->sock == NULL) goto failed; + if (composite_nomem(state->sock, result)) goto failed; + + if (my_address) { + state->my_address = talloc_strdup(state, my_address); + if (composite_nomem(state->my_address, result)) goto failed; + } state->my_port = my_port; + state->server_address = talloc_strdup(state, server_address); - if (state->sock == NULL) goto failed; + if (composite_nomem(state->server_address, result)) goto failed; + state->server_port = server_port; state->flags = flags; @@ -91,37 +134,13 @@ struct composite_context *socket_connect_send(struct socket_context *sock, return result; } - ctx = talloc_zero(state, struct composite_context); - if (ctx == NULL) goto failed; - ctx->state = COMPOSITE_STATE_IN_PROGRESS; - ctx->event_ctx = event_ctx; - ctx->async.fn = socket_connect_recv_conn; - ctx->async.private_data = state; - - state->ctx->status = socket_connect(sock, my_address, my_port, - server_address, server_port, - flags); - if (NT_STATUS_IS_ERR(state->ctx->status) && - !NT_STATUS_EQUAL(state->ctx->status, - NT_STATUS_MORE_PROCESSING_REQUIRED)) { - composite_trigger_error(state->ctx); - return result; - } - - state->connect_ev = event_add_fd(state->ctx->event_ctx, state, - socket_get_fd(sock), EVENT_FD_WRITE, - socket_connect_handler, ctx); - if (state->connect_ev == NULL) { - state->ctx->status = NT_STATUS_NO_MEMORY; - composite_trigger_error(state->ctx); - return result; - } + socket_send_connect(state); return result; - failed: - talloc_free(result); - return NULL; +failed: + composite_trigger_error(result); + return result; } /* @@ -143,6 +162,9 @@ static void socket_connect_handler(struct event_context *ev, composite_done(ctx); } +/* + recv name resolution reply then send the connect +*/ static void socket_connect_recv_addr(struct composite_context *ctx) { struct connect_state *state = @@ -152,35 +174,14 @@ static void socket_connect_recv_addr(struct composite_context *ctx) state->ctx->status = resolve_name_recv(ctx, state, &addr); if (!composite_is_ok(state->ctx)) return; - ctx = talloc_zero(state, struct composite_context); - if (composite_nomem(ctx, state->ctx)) return; - ctx->state = COMPOSITE_STATE_IN_PROGRESS; - ctx->event_ctx = state->ctx->event_ctx; - ctx->async.fn = socket_connect_recv_conn; - ctx->async.private_data = state; - - state->ctx->status = socket_connect(state->sock, - state->my_address, - state->my_port, - state->server_address, - state->server_port, - state->flags); - if (NT_STATUS_IS_ERR(state->ctx->status) && - !NT_STATUS_EQUAL(state->ctx->status, - NT_STATUS_MORE_PROCESSING_REQUIRED)) { - composite_error(state->ctx, state->ctx->status); - return; - } - - state->connect_ev = event_add_fd(ctx->event_ctx, state, - socket_get_fd(state->sock), - EVENT_FD_WRITE, - socket_connect_handler, ctx); - if (composite_nomem(state->connect_ev, state->ctx)) return; + state->server_address = addr; - return; + socket_send_connect(state); } +/* + called when a connect has finished. Complete the top level composite context +*/ static void socket_connect_recv_conn(struct composite_context *ctx) { struct connect_state *state = @@ -188,14 +189,13 @@ static void socket_connect_recv_conn(struct composite_context *ctx) state->ctx->status = ctx->status; if (!composite_is_ok(state->ctx)) return; - - /* We have to free the event context here because the continuation - * function might add an event for this socket directly. */ - talloc_free(state->connect_ev); - composite_done(state->ctx); } + +/* + wait for a socket_connect_send() to finish +*/ NTSTATUS socket_connect_recv(struct composite_context *ctx) { NTSTATUS status = composite_wait(ctx); @@ -203,6 +203,10 @@ NTSTATUS socket_connect_recv(struct composite_context *ctx) return status; } + +/* + like socket_connect() but takes an event context, doing a semi-async connect +*/ NTSTATUS socket_connect_ev(struct socket_context *sock, const char *my_address, int my_port, const char *server_address, int server_port, |