diff options
Diffstat (limited to '')
-rw-r--r-- | src/nxt_router.c | 407 |
1 files changed, 318 insertions, 89 deletions
diff --git a/src/nxt_router.c b/src/nxt_router.c index c71338fc..61daf746 100644 --- a/src/nxt_router.c +++ b/src/nxt_router.c @@ -48,14 +48,15 @@ typedef struct { struct nxt_req_app_link_s { uint32_t stream; + nxt_atomic_t use_count; nxt_port_t *app_port; - nxt_pid_t app_pid; nxt_port_t *reply_port; nxt_app_parse_ctx_t *ap; nxt_msg_info_t msg_info; nxt_req_conn_link_t *rc; - nxt_queue_link_t link; /* for nxt_app_t.requests */ + nxt_queue_link_t link_app_requests; /* for nxt_app_t.requests */ + nxt_queue_link_t link_port_pending; /* for nxt_port_t.pending_requests */ nxt_mp_t *mem_pool; nxt_work_t work; @@ -73,6 +74,24 @@ typedef struct { static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app); +nxt_inline void +nxt_router_ra_inc_use(nxt_req_app_link_t *ra) +{ + nxt_atomic_fetch_add(&ra->use_count, 1); +} + +nxt_inline void +nxt_router_ra_dec_use(nxt_req_app_link_t *ra) +{ + int c; + + c = nxt_atomic_fetch_add(&ra->use_count, -1); + + nxt_assert(c > 1); +} + +static void nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i); + static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task); static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data); static void nxt_router_conf_ready(nxt_task_t *task, @@ -153,6 +172,7 @@ static void nxt_router_app_port_error(nxt_task_t *task, static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app); static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port, uint32_t request_failed, uint32_t got_response); +static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra); static void nxt_router_conn_init(nxt_task_t *task, void *obj, void *data); static void nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj, @@ -162,7 +182,7 @@ static void nxt_router_conn_http_body_read(nxt_task_t *task, void *obj, void *data); static void nxt_router_process_http_request(nxt_task_t *task, nxt_conn_t *c, nxt_app_parse_ctx_t *ap); -static void nxt_router_process_http_request_mp(nxt_task_t *task, +static void nxt_router_app_prepare_request(nxt_task_t *task, nxt_req_app_link_t *ra); static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r, nxt_app_wmsg_t *wmsg); @@ -319,7 +339,7 @@ nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra, nxt_memzero(ra, sizeof(nxt_req_app_link_t)); ra->stream = rc->stream; - ra->app_pid = -1; + ra->use_count = 1; ra->rc = rc; rc->ra = ra; ra->reply_port = engine->port; @@ -338,6 +358,10 @@ nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src) nxt_mp_t *mp; nxt_req_app_link_t *ra; + if (ra_src->mem_pool != NULL) { + return ra_src; + } + mp = ra_src->ap->mem_pool; ra = nxt_mp_retain(mp, sizeof(nxt_req_app_link_t)); @@ -394,25 +418,38 @@ nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info, static void -nxt_router_ra_release(nxt_task_t *task, void *obj, void *data) +nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra); + + +static void +nxt_router_ra_update_peer_handler(nxt_task_t *task, void *obj, void *data) +{ + nxt_req_app_link_t *ra; + + ra = obj; + + nxt_router_ra_update_peer(task, ra); + + nxt_router_ra_use(task, ra, -1); +} + + +static void +nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra) { - nxt_req_app_link_t *ra; nxt_event_engine_t *engine; nxt_req_conn_link_t *rc; - ra = obj; - engine = data; + engine = ra->work.data; if (task->thread->engine != engine) { - if (ra->app_port != NULL) { - ra->app_pid = ra->app_port->pid; - } + nxt_router_ra_inc_use(ra); - ra->work.handler = nxt_router_ra_release; + ra->work.handler = nxt_router_ra_update_peer_handler; ra->work.task = &engine->task; ra->work.next = NULL; - nxt_debug(task, "ra stream #%uD post release to %p", + nxt_debug(task, "ra stream #%uD post update peer to %p", ra->stream, engine); nxt_event_engine_post(engine, &ra->work); @@ -420,22 +457,47 @@ nxt_router_ra_release(nxt_task_t *task, void *obj, void *data) return; } + nxt_debug(task, "ra stream #%uD update peer", ra->stream); + + rc = ra->rc; + + if (rc != NULL && ra->app_port != NULL) { + nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_port->pid); + } + + nxt_router_ra_use(task, ra, -1); +} + + +static void +nxt_router_ra_release(nxt_task_t *task, nxt_req_app_link_t *ra) +{ + nxt_conn_t *c; + nxt_req_conn_link_t *rc; + + nxt_assert(task->thread->engine == ra->work.data); + nxt_assert(ra->use_count == 0); + nxt_debug(task, "ra stream #%uD release", ra->stream); rc = ra->rc; if (rc != NULL) { - if (ra->app_pid != -1) { - nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_pid); - } + c = rc->conn; if (nxt_slow_path(ra->err_code != 0)) { - nxt_router_gen_error(task, rc->conn, ra->err_code, ra->err_str); + nxt_router_gen_error(task, c, ra->err_code, ra->err_str); } else { rc->app_port = ra->app_port; rc->msg_info = ra->msg_info; + if (rc->app->timeout != 0) { + c->read_timer.handler = nxt_router_app_timeout; + nxt_timer_add(task->thread->engine, &c->read_timer, + rc->app->timeout); + } + ra->app_port = NULL; ra->msg_info.buf = NULL; } @@ -458,6 +520,52 @@ nxt_router_ra_release(nxt_task_t *task, void *obj, void *data) } +static void +nxt_router_ra_release_handler(nxt_task_t *task, void *obj, void *data) +{ + nxt_req_app_link_t *ra; + + ra = obj; + + nxt_assert(ra->work.data == data); + + nxt_atomic_fetch_add(&ra->use_count, -1); + + nxt_router_ra_release(task, ra); +} + + +static void +nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i) +{ + int c; + nxt_event_engine_t *engine; + + c = nxt_atomic_fetch_add(&ra->use_count, i); + + if (i < 0 && c == -i) { + engine = ra->work.data; + + if (task->thread->engine == engine) { + nxt_router_ra_release(task, ra); + + return; + } + + nxt_router_ra_inc_use(ra); + + ra->work.handler = nxt_router_ra_release_handler; + ra->work.task = &engine->task; + ra->work.next = NULL; + + nxt_debug(task, "ra stream #%uD post release to %p", + ra->stream, engine); + + nxt_event_engine_post(engine, &ra->work); + } +} + + nxt_inline void nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str) { @@ -467,9 +575,25 @@ nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str) } +nxt_inline nxt_bool_t +nxt_queue_chk_remove(nxt_queue_link_t *lnk) +{ + if (lnk->next != NULL) { + nxt_queue_remove(lnk); + + lnk->next = NULL; + + return 1; + } + + return 0; +} + + nxt_inline void nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc) { + int ra_use_delta; nxt_req_app_link_t *ra; if (rc->app_port != NULL) { @@ -486,22 +610,25 @@ nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc) rc->ra = NULL; ra->rc = NULL; - nxt_thread_mutex_lock(&rc->app->mutex); + ra_use_delta = 0; - if (ra->link.next != NULL) { - nxt_queue_remove(&ra->link); + nxt_thread_mutex_lock(&rc->app->mutex); - ra->link.next = NULL; + if (ra->link_app_requests.next == NULL + && ra->link_port_pending.next == NULL) + { + ra = NULL; } else { - ra = NULL; + ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests); + ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending); } nxt_thread_mutex_unlock(&rc->app->mutex); - } - if (ra != NULL) { - nxt_router_ra_release(task, ra, ra->work.data); + if (ra != NULL) { + nxt_router_ra_use(task, ra, ra_use_delta); + } } if (rc->app != NULL) { @@ -2266,6 +2393,7 @@ nxt_router_response_ready_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg, size_t dump_size; nxt_buf_t *b, *last; nxt_conn_t *c; + nxt_event_engine_t *engine; nxt_req_conn_link_t *rc; b = msg->buf; @@ -2287,6 +2415,10 @@ nxt_router_response_ready_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg, b = NULL; } + engine = task->thread->engine; + + nxt_timer_disable(engine, &c->read_timer); + if (msg->port_msg.last != 0) { nxt_debug(task, "router data create last buf"); @@ -2298,6 +2430,12 @@ nxt_router_response_ready_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg, nxt_buf_chain_add(&b, last); nxt_router_rc_unlink(task, rc); + + } else { + if (rc->app->timeout != 0) { + c->read_timer.handler = nxt_router_app_timeout; + nxt_timer_add(engine, &c->read_timer, rc->app->timeout); + } } if (b == NULL) { @@ -2327,10 +2465,41 @@ static void nxt_router_response_error_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg, void *data) { + nxt_int_t res; + nxt_port_t *port; + nxt_bool_t cancelled; + nxt_req_app_link_t *ra; nxt_req_conn_link_t *rc; rc = data; + ra = rc->ra; + + if (ra != NULL) { + cancelled = nxt_router_msg_cancel(task, &ra->msg_info, ra->stream); + + if (cancelled) { + nxt_router_ra_inc_use(ra); + + res = nxt_router_app_port(task, ra); + + if (res == NXT_OK) { + port = ra->app_port; + + nxt_assert(port != NULL); + + nxt_port_rpc_ex_set_peer(task, task->thread->engine->port, rc, + port->pid); + + nxt_router_app_prepare_request(task, ra); + } + + msg->port_msg.last = 0; + + return; + } + } + nxt_router_gen_error(task, rc->conn, 500, "Application terminated unexpectedly"); @@ -2478,7 +2647,7 @@ nxt_router_app_port_error(nxt_task_t *task, nxt_port_recv_msg_t *msg, nxt_queue_remove(lnk); lnk->next = NULL; - ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link); + ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link_app_requests); } else { ra = NULL; @@ -2491,7 +2660,7 @@ nxt_router_app_port_error(nxt_task_t *task, nxt_port_recv_msg_t *msg, &app->name, app, ra->stream); nxt_router_ra_error(ra, 500, "Failed to start application worker"); - nxt_router_ra_release(task, ra, ra->work.data); + nxt_router_ra_use(task, ra, -1); } nxt_router_app_use(task, app, -1); @@ -2533,7 +2702,7 @@ nxt_router_app_first_port_busy(nxt_app_t *app) nxt_inline nxt_port_t * -nxt_router_app_get_port_unsafe(nxt_app_t *app, int *use_delta) +nxt_router_app_get_port_unsafe(nxt_app_t *app) { nxt_port_t *port; nxt_queue_link_t *lnk; @@ -2550,10 +2719,10 @@ nxt_router_app_get_port_unsafe(nxt_app_t *app, int *use_delta) { nxt_queue_insert_tail(&app->ports, lnk); + nxt_port_inc_use(port); + } else { lnk->next = NULL; - - (*use_delta)--; } return port; @@ -2606,7 +2775,7 @@ nxt_router_app_process_request(nxt_task_t *task, void *obj, void *data) nxt_debug(task, "app '%V' %p process next stream #%uD", &app->name, app, ra->stream); - nxt_router_process_http_request_mp(task, ra); + nxt_router_app_prepare_request(task, ra); } @@ -2614,19 +2783,16 @@ static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port, uint32_t request_failed, uint32_t got_response) { - int use_delta, ra_use_delta; nxt_app_t *app; nxt_bool_t send_quit; nxt_queue_link_t *lnk; - nxt_req_app_link_t *ra; + nxt_req_app_link_t *ra, *next_ra; nxt_assert(port != NULL); nxt_assert(port->app != NULL); app = port->app; - use_delta = (request_failed == 0 && got_response == 0) ? 0 : -1; - nxt_thread_mutex_lock(&app->mutex); port->app_pending_responses -= request_failed + got_response; @@ -2645,7 +2811,7 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port, nxt_queue_insert_head(&app->ports, &port->app_link); } - use_delta++; + nxt_port_inc_use(port); } else { if (port->app_pending_responses == 0 @@ -2665,21 +2831,46 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port, nxt_queue_remove(lnk); lnk->next = NULL; - ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link); + ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, link_app_requests); - ra_use_delta = 1; - ra->app_port = nxt_router_app_get_port_unsafe(app, &ra_use_delta); + ra->app_port = nxt_router_app_get_port_unsafe(app); + + if (ra->app_port->app_pending_responses > 1) { + nxt_queue_insert_tail(&ra->app_port->pending_requests, + &ra->link_port_pending); + + nxt_router_ra_inc_use(ra); + } } else { ra = NULL; - ra_use_delta = 0; + } + + if ((request_failed > 0 || got_response > 0) + && !nxt_queue_is_empty(&port->pending_requests)) + { + lnk = nxt_queue_first(&port->pending_requests); + nxt_queue_remove(lnk); + lnk->next = NULL; + + next_ra = nxt_queue_link_data(lnk, nxt_req_app_link_t, + link_port_pending); + + } else { + next_ra = NULL; } send_quit = app->live == 0 && port->app_pending_responses > 0; nxt_thread_mutex_unlock(&app->mutex); + if (next_ra != NULL) { + nxt_router_ra_use(task, next_ra, -1); + } + if (ra != NULL) { + nxt_router_ra_use(task, ra, -1); + nxt_work_queue_add(&task->thread->engine->fast_work_queue, nxt_router_app_process_request, &task->thread->engine->task, app, ra); @@ -2710,12 +2901,8 @@ nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port, adjust_use: - if (use_delta != 0) { - nxt_port_use(task, port, use_delta); - } - - if (ra_use_delta != 0) { - nxt_port_use(task, ra->app_port, ra_use_delta); + if (request_failed > 0 || got_response > 0) { + nxt_port_use(task, port, -1); } } @@ -2766,41 +2953,46 @@ nxt_router_app_port_close(nxt_task_t *task, nxt_port_t *port) static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra) { - int use_delta; - nxt_int_t res; - nxt_app_t *app; - nxt_bool_t can_start_worker; - nxt_conn_t *c; - nxt_port_t *port; - nxt_event_engine_t *engine; - nxt_socket_conf_joint_t *joint; + int failed_port_use_delta; + nxt_int_t res; + nxt_app_t *app; + nxt_bool_t can_start_worker; + nxt_conn_t *c; + nxt_port_t *port, *failed_port; - use_delta = 1; c = ra->rc->conn; + app = ra->rc->app; - joint = c->joint; - app = joint->socket_conf->application; + failed_port_use_delta = 0; - if (app == NULL) { - nxt_router_gen_error(task, c, 500, - "Application is NULL in socket_conf"); - return NXT_ERROR; + nxt_thread_mutex_lock(&app->mutex); + + if (nxt_queue_chk_remove(&ra->link_app_requests)) + { + nxt_router_ra_dec_use(ra); } - ra->rc->app = app; + if (nxt_queue_chk_remove(&ra->link_port_pending)) + { + nxt_router_ra_dec_use(ra); + } - nxt_router_app_use(task, app, 1); + if (ra->app_port != NULL) { + failed_port = ra->app_port; + failed_port_use_delta--; - engine = task->thread->engine; + failed_port->app_pending_responses--; - nxt_timer_disable(engine, &c->read_timer); + if (failed_port->app_link.next != NULL) { + nxt_queue_remove(&failed_port->app_link); + failed_port->app_link.next = NULL; - if (app->timeout != 0) { - c->read_timer.handler = nxt_router_app_timeout; - nxt_timer_add(engine, &c->read_timer, app->timeout); - } + failed_port_use_delta--; + } - nxt_thread_mutex_lock(&app->mutex); + } else { + failed_port = NULL; + } can_start_worker = (app->workers + app->pending_workers) < app->max_workers; @@ -2811,7 +3003,12 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra) ra = nxt_router_ra_create(task, ra); if (nxt_fast_path(ra != NULL)) { - nxt_queue_insert_tail(&app->requests, &ra->link); + nxt_queue_insert_tail(&app->requests, &ra->link_app_requests); + + nxt_router_ra_inc_use(ra); + + nxt_debug(task, "ra stream #%uD enqueue to app->requests", + ra->stream); if (can_start_worker) { app->pending_workers++; @@ -2821,14 +3018,37 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra) port = NULL; } else { - port = nxt_router_app_get_port_unsafe(app, &use_delta); + port = nxt_router_app_get_port_unsafe(app); + + if (port->app_pending_responses > 1) { + ra = nxt_router_ra_create(task, ra); + + if (nxt_fast_path(ra != NULL)) { + nxt_queue_insert_tail(&port->pending_requests, + &ra->link_port_pending); + + nxt_router_ra_inc_use(ra); + + nxt_debug(task, "ra stream #%uD enqueue to " + "port->pending_requests", ra->stream); + } + } } nxt_thread_mutex_unlock(&app->mutex); + if (failed_port_use_delta != 0) { + nxt_port_use(task, failed_port, failed_port_use_delta); + } + if (nxt_slow_path(ra == NULL)) { nxt_router_gen_error(task, c, 500, "Failed to allocate " "req<->app link"); + + if (port != NULL) { + nxt_port_use(task, port, -1); + } + return NXT_ERROR; } @@ -2837,14 +3057,9 @@ nxt_router_app_port(nxt_task_t *task, nxt_req_app_link_t *ra) ra->app_port = port; - if (use_delta != 0) { - nxt_port_use(task, port, use_delta); - } return NXT_OK; } - nxt_debug(task, "ra stream #%uD allocated", ra->stream); - if (!can_start_worker) { nxt_debug(task, "app '%V' %p too many running or pending workers", &app->name, app); @@ -3127,11 +3342,22 @@ static void nxt_router_process_http_request(nxt_task_t *task, nxt_conn_t *c, nxt_app_parse_ctx_t *ap) { - nxt_int_t res; - nxt_port_t *port; - nxt_event_engine_t *engine; - nxt_req_app_link_t ra_local, *ra; - nxt_req_conn_link_t *rc; + nxt_int_t res; + nxt_app_t *app; + nxt_port_t *port; + nxt_event_engine_t *engine; + nxt_req_app_link_t ra_local, *ra; + nxt_req_conn_link_t *rc; + nxt_socket_conf_joint_t *joint; + + joint = c->joint; + app = joint->socket_conf->application; + + if (app == NULL) { + nxt_router_gen_error(task, c, 500, + "Application is NULL in socket_conf"); + return; + } engine = task->thread->engine; @@ -3149,6 +3375,11 @@ nxt_router_process_http_request(nxt_task_t *task, nxt_conn_t *c, rc->stream = nxt_port_rpc_ex_stream(rc); rc->conn = c; + rc->app = app; + + nxt_router_app_use(task, app, 1); + + nxt_timer_disable(engine, &c->read_timer); nxt_queue_insert_tail(&c->requests, &rc->link); @@ -3167,16 +3398,14 @@ nxt_router_process_http_request(nxt_task_t *task, nxt_conn_t *c, return; } + ra = rc->ra; port = ra->app_port; - if (nxt_slow_path(port == NULL)) { - nxt_router_gen_error(task, c, 500, "Application port not found"); - return; - } + nxt_assert(port != NULL); nxt_port_rpc_ex_set_peer(task, engine->port, rc, port->pid); - nxt_router_process_http_request_mp(task, ra); + nxt_router_app_prepare_request(task, ra); } @@ -3187,7 +3416,7 @@ nxt_router_dummy_buf_completion(nxt_task_t *task, void *obj, void *data) static void -nxt_router_process_http_request_mp(nxt_task_t *task, nxt_req_app_link_t *ra) +nxt_router_app_prepare_request(nxt_task_t *task, nxt_req_app_link_t *ra) { uint32_t request_failed; nxt_buf_t *b; @@ -3266,7 +3495,7 @@ release_port: nxt_router_app_port_release(task, port, request_failed, 0); - nxt_router_ra_release(task, ra, ra->work.data); + nxt_router_ra_update_peer(task, ra); } |