xref: /unit/src/nxt_router.c (revision 648)
120Sigor@sysoev.ru 
220Sigor@sysoev.ru /*
320Sigor@sysoev.ru  * Copyright (C) Igor Sysoev
420Sigor@sysoev.ru  * Copyright (C) Valentin V. Bartenev
520Sigor@sysoev.ru  * Copyright (C) NGINX, Inc.
620Sigor@sysoev.ru  */
720Sigor@sysoev.ru 
853Sigor@sysoev.ru #include <nxt_router.h>
9115Sigor@sysoev.ru #include <nxt_conf.h>
10431Sigor@sysoev.ru #include <nxt_http.h>
1120Sigor@sysoev.ru 
1220Sigor@sysoev.ru 
13115Sigor@sysoev.ru typedef struct {
14318Smax.romanov@nginx.com     nxt_str_t         type;
15507Smax.romanov@nginx.com     uint32_t          processes;
16507Smax.romanov@nginx.com     uint32_t          max_processes;
17507Smax.romanov@nginx.com     uint32_t          spare_processes;
18318Smax.romanov@nginx.com     nxt_msec_t        timeout;
19427Smax.romanov@nginx.com     nxt_msec_t        res_timeout;
20507Smax.romanov@nginx.com     nxt_msec_t        idle_timeout;
21318Smax.romanov@nginx.com     uint32_t          requests;
22318Smax.romanov@nginx.com     nxt_conf_value_t  *limits_value;
23507Smax.romanov@nginx.com     nxt_conf_value_t  *processes_value;
24133Sigor@sysoev.ru } nxt_router_app_conf_t;
25133Sigor@sysoev.ru 
26133Sigor@sysoev.ru 
27133Sigor@sysoev.ru typedef struct {
28133Sigor@sysoev.ru     nxt_str_t  application;
29115Sigor@sysoev.ru } nxt_router_listener_conf_t;
30115Sigor@sysoev.ru 
31115Sigor@sysoev.ru 
32423Smax.romanov@nginx.com typedef struct nxt_msg_info_s {
33423Smax.romanov@nginx.com     nxt_buf_t                 *buf;
34423Smax.romanov@nginx.com     nxt_port_mmap_tracking_t  tracking;
35423Smax.romanov@nginx.com     nxt_work_handler_t        completion_handler;
36423Smax.romanov@nginx.com } nxt_msg_info_t;
37423Smax.romanov@nginx.com 
38423Smax.romanov@nginx.com 
39167Smax.romanov@nginx.com typedef struct nxt_req_app_link_s nxt_req_app_link_t;
40141Smax.romanov@nginx.com 
41141Smax.romanov@nginx.com 
42318Smax.romanov@nginx.com typedef struct {
43431Sigor@sysoev.ru     uint32_t                 stream;
44431Sigor@sysoev.ru     nxt_app_t                *app;
45431Sigor@sysoev.ru     nxt_port_t               *app_port;
46431Sigor@sysoev.ru     nxt_app_parse_ctx_t      *ap;
47431Sigor@sysoev.ru     nxt_msg_info_t           msg_info;
48431Sigor@sysoev.ru     nxt_req_app_link_t       *ra;
49431Sigor@sysoev.ru 
50431Sigor@sysoev.ru     nxt_queue_link_t         link;     /* for nxt_conn_t.requests */
51318Smax.romanov@nginx.com } nxt_req_conn_link_t;
52318Smax.romanov@nginx.com 
53318Smax.romanov@nginx.com 
54167Smax.romanov@nginx.com struct nxt_req_app_link_s {
55318Smax.romanov@nginx.com     uint32_t             stream;
56425Smax.romanov@nginx.com     nxt_atomic_t         use_count;
57167Smax.romanov@nginx.com     nxt_port_t           *app_port;
58167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
59167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
60423Smax.romanov@nginx.com     nxt_msg_info_t       msg_info;
61167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
62167Smax.romanov@nginx.com 
63427Smax.romanov@nginx.com     nxt_nsec_t           res_time;
64427Smax.romanov@nginx.com 
65425Smax.romanov@nginx.com     nxt_queue_link_t     link_app_requests; /* for nxt_app_t.requests */
66425Smax.romanov@nginx.com     nxt_queue_link_t     link_port_pending; /* for nxt_port_t.pending_requests */
67427Smax.romanov@nginx.com     nxt_queue_link_t     link_app_pending;  /* for nxt_app_t.pending */
68167Smax.romanov@nginx.com 
69167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
70167Smax.romanov@nginx.com     nxt_work_t           work;
71345Smax.romanov@nginx.com 
72345Smax.romanov@nginx.com     int                  err_code;
73345Smax.romanov@nginx.com     const char           *err_str;
74167Smax.romanov@nginx.com };
75167Smax.romanov@nginx.com 
76167Smax.romanov@nginx.com 
77198Sigor@sysoev.ru typedef struct {
78198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
79198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
80198Sigor@sysoev.ru } nxt_socket_rpc_t;
81198Sigor@sysoev.ru 
82198Sigor@sysoev.ru 
83507Smax.romanov@nginx.com typedef struct {
84507Smax.romanov@nginx.com     nxt_app_t               *app;
85507Smax.romanov@nginx.com     nxt_router_temp_conf_t  *temp_conf;
86507Smax.romanov@nginx.com } nxt_app_rpc_t;
87507Smax.romanov@nginx.com 
88507Smax.romanov@nginx.com 
89427Smax.romanov@nginx.com struct nxt_port_select_state_s {
90427Smax.romanov@nginx.com     nxt_app_t           *app;
91427Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
92427Smax.romanov@nginx.com 
93427Smax.romanov@nginx.com     nxt_port_t          *failed_port;
94427Smax.romanov@nginx.com     int                 failed_port_use_delta;
95427Smax.romanov@nginx.com 
96507Smax.romanov@nginx.com     uint8_t             start_process;    /* 1 bit */
97427Smax.romanov@nginx.com     nxt_req_app_link_t  *shared_ra;
98427Smax.romanov@nginx.com     nxt_port_t          *port;
99427Smax.romanov@nginx.com };
100427Smax.romanov@nginx.com 
101427Smax.romanov@nginx.com typedef struct nxt_port_select_state_s nxt_port_select_state_t;
102427Smax.romanov@nginx.com 
103427Smax.romanov@nginx.com static void nxt_router_port_select(nxt_task_t *task,
104427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
105427Smax.romanov@nginx.com 
106427Smax.romanov@nginx.com static nxt_int_t nxt_router_port_post_select(nxt_task_t *task,
107427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
108427Smax.romanov@nginx.com 
109507Smax.romanov@nginx.com static nxt_int_t nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app);
110343Smax.romanov@nginx.com 
111425Smax.romanov@nginx.com nxt_inline void
112425Smax.romanov@nginx.com nxt_router_ra_inc_use(nxt_req_app_link_t *ra)
113425Smax.romanov@nginx.com {
114425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, 1);
115425Smax.romanov@nginx.com }
116425Smax.romanov@nginx.com 
117425Smax.romanov@nginx.com nxt_inline void
118425Smax.romanov@nginx.com nxt_router_ra_dec_use(nxt_req_app_link_t *ra)
119425Smax.romanov@nginx.com {
120538Svbart@nginx.com #if (NXT_DEBUG)
121425Smax.romanov@nginx.com     int  c;
122425Smax.romanov@nginx.com 
123425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, -1);
124425Smax.romanov@nginx.com 
125425Smax.romanov@nginx.com     nxt_assert(c > 1);
126538Svbart@nginx.com #else
127538Svbart@nginx.com     (void) nxt_atomic_fetch_add(&ra->use_count, -1);
128538Svbart@nginx.com #endif
129425Smax.romanov@nginx.com }
130425Smax.romanov@nginx.com 
131425Smax.romanov@nginx.com static void nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i);
132425Smax.romanov@nginx.com 
133139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
134198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
135198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
136139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
137139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
138139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
139139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
140193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
14153Sigor@sysoev.ru 
142115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
143115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
144133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
145133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
146133Sigor@sysoev.ru     nxt_str_t *name);
147198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
148198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
149198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
150198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
151198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
152198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
153507Smax.romanov@nginx.com static void nxt_router_app_rpc_create(nxt_task_t *task,
154507Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_app_t *app);
155507Smax.romanov@nginx.com static void nxt_router_app_prefork_ready(nxt_task_t *task,
156507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
157507Smax.romanov@nginx.com static void nxt_router_app_prefork_error(nxt_task_t *task,
158507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
159359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
160359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
161359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
162359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
16353Sigor@sysoev.ru 
16453Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
16553Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
16653Sigor@sysoev.ru     const nxt_event_interface_t *interface);
167115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
168115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
169115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
170115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
171115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
172115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
173154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
174154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
175154Sigor@sysoev.ru     nxt_work_handler_t handler);
176313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
177313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
178139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
179139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
18053Sigor@sysoev.ru 
18153Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
18253Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
18353Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
18453Sigor@sysoev.ru     nxt_event_engine_t *engine);
185343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
186133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
18753Sigor@sysoev.ru 
188315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
189315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
190315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
191315Sigor@sysoev.ru     nxt_work_t *jobs);
19253Sigor@sysoev.ru 
19353Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
19453Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
19553Sigor@sysoev.ru     void *data);
19653Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
19753Sigor@sysoev.ru     void *data);
19853Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
19953Sigor@sysoev.ru     void *data);
200313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
201313Sigor@sysoev.ru     void *data);
20253Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
20353Sigor@sysoev.ru     void *data);
20453Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
20553Sigor@sysoev.ru     void *data);
206359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
207359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
20853Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
20953Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
21053Sigor@sysoev.ru 
211630Svbart@nginx.com static void nxt_router_access_log_writer(nxt_task_t *task,
212630Svbart@nginx.com     nxt_http_request_t *r, nxt_router_access_log_t *access_log);
213630Svbart@nginx.com static u_char *nxt_router_access_log_date(u_char *buf, nxt_realtime_t *now,
214630Svbart@nginx.com     struct tm *tm, size_t size, const char *format);
215630Svbart@nginx.com static void nxt_router_access_log_open(nxt_task_t *task,
216630Svbart@nginx.com     nxt_router_temp_conf_t *tmcf);
217630Svbart@nginx.com static void nxt_router_access_log_ready(nxt_task_t *task,
218630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
219630Svbart@nginx.com static void nxt_router_access_log_error(nxt_task_t *task,
220630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
221630Svbart@nginx.com static void nxt_router_access_log_release(nxt_task_t *task,
222630Svbart@nginx.com     nxt_thread_spinlock_t *lock, nxt_router_access_log_t *access_log);
223631Svbart@nginx.com static void nxt_router_access_log_reopen_ready(nxt_task_t *task,
224631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
225631Svbart@nginx.com static void nxt_router_access_log_reopen_error(nxt_task_t *task,
226631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
227630Svbart@nginx.com 
228343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
229343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
230343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
231343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
232343Smax.romanov@nginx.com 
233507Smax.romanov@nginx.com static void nxt_router_app_quit(nxt_task_t *task, nxt_app_t *app);
234343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
235343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
236427Smax.romanov@nginx.com static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_app_t *app,
237427Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
238141Smax.romanov@nginx.com 
239425Smax.romanov@nginx.com static void nxt_router_app_prepare_request(nxt_task_t *task,
240343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
241216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
242216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
243216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
244216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
245216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
246216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
247510Salexander.borisov@nginx.com static nxt_int_t nxt_perl_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
248510Salexander.borisov@nginx.com     nxt_app_wmsg_t *wmsg);
249584Salexander.borisov@nginx.com static nxt_int_t nxt_ruby_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
250584Salexander.borisov@nginx.com     nxt_app_wmsg_t *wmsg);
251510Salexander.borisov@nginx.com 
25253Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
253318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
254507Smax.romanov@nginx.com static void nxt_router_adjust_idle_timer(nxt_task_t *task, void *obj,
255507Smax.romanov@nginx.com     void *data);
256507Smax.romanov@nginx.com static void nxt_router_app_idle_timeout(nxt_task_t *task, void *obj,
257507Smax.romanov@nginx.com     void *data);
258507Smax.romanov@nginx.com static void nxt_router_app_release_handler(nxt_task_t *task, void *obj,
259507Smax.romanov@nginx.com     void *data);
260431Sigor@sysoev.ru 
261431Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_request_send_state;
262431Sigor@sysoev.ru static void nxt_http_request_send_body(nxt_task_t *task, void *obj, void *data);
263141Smax.romanov@nginx.com 
264119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
26520Sigor@sysoev.ru 
266216Sigor@sysoev.ru 
267216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
268216Sigor@sysoev.ru     nxt_python_prepare_msg,
269216Sigor@sysoev.ru     nxt_php_prepare_msg,
270216Sigor@sysoev.ru     nxt_go_prepare_msg,
271510Salexander.borisov@nginx.com     nxt_perl_prepare_msg,
272584Salexander.borisov@nginx.com     nxt_ruby_prepare_msg,
273216Sigor@sysoev.ru };
274216Sigor@sysoev.ru 
275216Sigor@sysoev.ru 
27620Sigor@sysoev.ru nxt_int_t
277141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
27820Sigor@sysoev.ru {
279141Smax.romanov@nginx.com     nxt_int_t      ret;
280141Smax.romanov@nginx.com     nxt_router_t   *router;
281141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
282141Smax.romanov@nginx.com 
283141Smax.romanov@nginx.com     rt = task->thread->runtime;
28453Sigor@sysoev.ru 
285431Sigor@sysoev.ru     ret = nxt_http_init(task, rt);
28688Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
28788Smax.romanov@nginx.com         return ret;
28888Smax.romanov@nginx.com     }
28988Smax.romanov@nginx.com 
29053Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
29153Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
29253Sigor@sysoev.ru         return NXT_ERROR;
29353Sigor@sysoev.ru     }
29453Sigor@sysoev.ru 
29553Sigor@sysoev.ru     nxt_queue_init(&router->engines);
29653Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
297133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
29853Sigor@sysoev.ru 
299119Smax.romanov@nginx.com     nxt_router = router;
300119Smax.romanov@nginx.com 
301115Sigor@sysoev.ru     return NXT_OK;
302115Sigor@sysoev.ru }
303115Sigor@sysoev.ru 
304115Sigor@sysoev.ru 
305343Smax.romanov@nginx.com static void
306507Smax.romanov@nginx.com nxt_router_start_app_process_handler(nxt_task_t *task, nxt_port_t *port,
307507Smax.romanov@nginx.com     void *data)
308167Smax.romanov@nginx.com {
309343Smax.romanov@nginx.com     size_t         size;
310343Smax.romanov@nginx.com     uint32_t       stream;
311430Sigor@sysoev.ru     nxt_mp_t       *mp;
312*648Svbart@nginx.com     nxt_int_t      ret;
313343Smax.romanov@nginx.com     nxt_app_t      *app;
314343Smax.romanov@nginx.com     nxt_buf_t      *b;
315343Smax.romanov@nginx.com     nxt_port_t     *main_port;
316343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
317343Smax.romanov@nginx.com 
318343Smax.romanov@nginx.com     app = data;
319167Smax.romanov@nginx.com 
320167Smax.romanov@nginx.com     rt = task->thread->runtime;
321240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
322167Smax.romanov@nginx.com 
323507Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start process", &app->name, app);
324343Smax.romanov@nginx.com 
325343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
326343Smax.romanov@nginx.com 
327343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
328343Smax.romanov@nginx.com 
329343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
330343Smax.romanov@nginx.com         goto failed;
331167Smax.romanov@nginx.com     }
332167Smax.romanov@nginx.com 
333343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
334343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
335343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
336343Smax.romanov@nginx.com 
337343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
338343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
339343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
340343Smax.romanov@nginx.com                                            -1, app);
341343Smax.romanov@nginx.com 
342343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
343343Smax.romanov@nginx.com         goto failed;
344343Smax.romanov@nginx.com     }
345343Smax.romanov@nginx.com 
346*648Svbart@nginx.com     ret = nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
347*648Svbart@nginx.com                                 stream, port->id, b);
348*648Svbart@nginx.com 
349*648Svbart@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
350*648Svbart@nginx.com         nxt_port_rpc_cancel(task, port, stream);
351*648Svbart@nginx.com         goto failed;
352*648Svbart@nginx.com     }
353343Smax.romanov@nginx.com 
354343Smax.romanov@nginx.com     return;
355343Smax.romanov@nginx.com 
356343Smax.romanov@nginx.com failed:
357343Smax.romanov@nginx.com 
358*648Svbart@nginx.com     if (b != NULL) {
359*648Svbart@nginx.com         mp = b->data;
360*648Svbart@nginx.com         nxt_mp_free(mp, b);
361*648Svbart@nginx.com         nxt_mp_release(mp);
362*648Svbart@nginx.com     }
363*648Svbart@nginx.com 
364343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
365343Smax.romanov@nginx.com 
366507Smax.romanov@nginx.com     app->pending_processes--;
367343Smax.romanov@nginx.com 
368343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
369343Smax.romanov@nginx.com 
370343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
371167Smax.romanov@nginx.com }
372167Smax.romanov@nginx.com 
373167Smax.romanov@nginx.com 
374343Smax.romanov@nginx.com static nxt_int_t
375507Smax.romanov@nginx.com nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app)
376141Smax.romanov@nginx.com {
377343Smax.romanov@nginx.com     nxt_int_t      res;
378343Smax.romanov@nginx.com     nxt_port_t     *router_port;
379343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
380343Smax.romanov@nginx.com 
381343Smax.romanov@nginx.com     rt = task->thread->runtime;
382343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
383343Smax.romanov@nginx.com 
384343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
385343Smax.romanov@nginx.com 
386507Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_app_process_handler,
387343Smax.romanov@nginx.com                         app);
388343Smax.romanov@nginx.com 
389343Smax.romanov@nginx.com     if (res == NXT_OK) {
390343Smax.romanov@nginx.com         return res;
391318Smax.romanov@nginx.com     }
392318Smax.romanov@nginx.com 
393343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
394343Smax.romanov@nginx.com 
395507Smax.romanov@nginx.com     app->pending_processes--;
396343Smax.romanov@nginx.com 
397343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
398343Smax.romanov@nginx.com 
399343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
400343Smax.romanov@nginx.com 
401343Smax.romanov@nginx.com     return NXT_ERROR;
402318Smax.romanov@nginx.com }
403318Smax.romanov@nginx.com 
404318Smax.romanov@nginx.com 
405351Smax.romanov@nginx.com nxt_inline void
406351Smax.romanov@nginx.com nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra,
407351Smax.romanov@nginx.com     nxt_req_conn_link_t *rc)
408167Smax.romanov@nginx.com {
409318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
410351Smax.romanov@nginx.com 
411318Smax.romanov@nginx.com     engine = task->thread->engine;
412167Smax.romanov@nginx.com 
413167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
414167Smax.romanov@nginx.com 
415318Smax.romanov@nginx.com     ra->stream = rc->stream;
416425Smax.romanov@nginx.com     ra->use_count = 1;
417167Smax.romanov@nginx.com     ra->rc = rc;
418318Smax.romanov@nginx.com     rc->ra = ra;
419318Smax.romanov@nginx.com     ra->reply_port = engine->port;
420351Smax.romanov@nginx.com     ra->ap = rc->ap;
421167Smax.romanov@nginx.com 
422167Smax.romanov@nginx.com     ra->work.handler = NULL;
423318Smax.romanov@nginx.com     ra->work.task = &engine->task;
424167Smax.romanov@nginx.com     ra->work.obj = ra;
425318Smax.romanov@nginx.com     ra->work.data = engine;
426351Smax.romanov@nginx.com }
427351Smax.romanov@nginx.com 
428351Smax.romanov@nginx.com 
429351Smax.romanov@nginx.com nxt_inline nxt_req_app_link_t *
430351Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src)
431351Smax.romanov@nginx.com {
432351Smax.romanov@nginx.com     nxt_mp_t            *mp;
433351Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
434351Smax.romanov@nginx.com 
435425Smax.romanov@nginx.com     if (ra_src->mem_pool != NULL) {
436425Smax.romanov@nginx.com         return ra_src;
437425Smax.romanov@nginx.com     }
438425Smax.romanov@nginx.com 
439351Smax.romanov@nginx.com     mp = ra_src->ap->mem_pool;
440351Smax.romanov@nginx.com 
441430Sigor@sysoev.ru     ra = nxt_mp_alloc(mp, sizeof(nxt_req_app_link_t));
442351Smax.romanov@nginx.com 
443351Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
444351Smax.romanov@nginx.com 
445351Smax.romanov@nginx.com         ra_src->rc->ra = NULL;
446351Smax.romanov@nginx.com         ra_src->rc = NULL;
447351Smax.romanov@nginx.com 
448351Smax.romanov@nginx.com         return NULL;
449351Smax.romanov@nginx.com     }
450351Smax.romanov@nginx.com 
451430Sigor@sysoev.ru     nxt_mp_retain(mp);
452430Sigor@sysoev.ru 
453351Smax.romanov@nginx.com     nxt_router_ra_init(task, ra, ra_src->rc);
454351Smax.romanov@nginx.com 
455351Smax.romanov@nginx.com     ra->mem_pool = mp;
456167Smax.romanov@nginx.com 
457167Smax.romanov@nginx.com     return ra;
458167Smax.romanov@nginx.com }
459167Smax.romanov@nginx.com 
460167Smax.romanov@nginx.com 
461423Smax.romanov@nginx.com nxt_inline nxt_bool_t
462423Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info,
463423Smax.romanov@nginx.com     uint32_t stream)
464423Smax.romanov@nginx.com {
465423Smax.romanov@nginx.com     nxt_buf_t   *b, *next;
466423Smax.romanov@nginx.com     nxt_bool_t  cancelled;
467423Smax.romanov@nginx.com 
468423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
469423Smax.romanov@nginx.com         return 0;
470423Smax.romanov@nginx.com     }
471423Smax.romanov@nginx.com 
472423Smax.romanov@nginx.com     cancelled = nxt_port_mmap_tracking_cancel(task, &msg_info->tracking,
473423Smax.romanov@nginx.com                                               stream);
474423Smax.romanov@nginx.com 
475423Smax.romanov@nginx.com     if (cancelled) {
476423Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router", stream);
477423Smax.romanov@nginx.com     }
478423Smax.romanov@nginx.com 
479423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
480423Smax.romanov@nginx.com         next = b->next;
481423Smax.romanov@nginx.com 
482423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
483423Smax.romanov@nginx.com 
484423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
485423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
486423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
487423Smax.romanov@nginx.com         }
488423Smax.romanov@nginx.com     }
489423Smax.romanov@nginx.com 
490423Smax.romanov@nginx.com     msg_info->buf = NULL;
491423Smax.romanov@nginx.com 
492423Smax.romanov@nginx.com     return cancelled;
493423Smax.romanov@nginx.com }
494423Smax.romanov@nginx.com 
495423Smax.romanov@nginx.com 
496167Smax.romanov@nginx.com static void
497425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra);
498425Smax.romanov@nginx.com 
499425Smax.romanov@nginx.com 
500425Smax.romanov@nginx.com static void
501425Smax.romanov@nginx.com nxt_router_ra_update_peer_handler(nxt_task_t *task, void *obj, void *data)
502167Smax.romanov@nginx.com {
503425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
504425Smax.romanov@nginx.com 
505425Smax.romanov@nginx.com     ra = obj;
506425Smax.romanov@nginx.com 
507425Smax.romanov@nginx.com     nxt_router_ra_update_peer(task, ra);
508425Smax.romanov@nginx.com 
509425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
510425Smax.romanov@nginx.com }
511425Smax.romanov@nginx.com 
512425Smax.romanov@nginx.com 
513425Smax.romanov@nginx.com static void
514425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra)
515425Smax.romanov@nginx.com {
516343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
517343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
518318Smax.romanov@nginx.com 
519425Smax.romanov@nginx.com     engine = ra->work.data;
520318Smax.romanov@nginx.com 
521343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
522425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
523425Smax.romanov@nginx.com 
524425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_update_peer_handler;
525318Smax.romanov@nginx.com         ra->work.task = &engine->task;
526318Smax.romanov@nginx.com         ra->work.next = NULL;
527318Smax.romanov@nginx.com 
528425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post update peer to %p",
529318Smax.romanov@nginx.com                   ra->stream, engine);
530318Smax.romanov@nginx.com 
531318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
532318Smax.romanov@nginx.com 
533318Smax.romanov@nginx.com         return;
534318Smax.romanov@nginx.com     }
535318Smax.romanov@nginx.com 
536425Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD update peer", ra->stream);
537425Smax.romanov@nginx.com 
538425Smax.romanov@nginx.com     rc = ra->rc;
539425Smax.romanov@nginx.com 
540425Smax.romanov@nginx.com     if (rc != NULL && ra->app_port != NULL) {
541425Smax.romanov@nginx.com         nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_port->pid);
542425Smax.romanov@nginx.com     }
543425Smax.romanov@nginx.com 
544425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
545425Smax.romanov@nginx.com }
546425Smax.romanov@nginx.com 
547425Smax.romanov@nginx.com 
548425Smax.romanov@nginx.com static void
549425Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, nxt_req_app_link_t *ra)
550425Smax.romanov@nginx.com {
551431Sigor@sysoev.ru     nxt_mp_t                *mp;
552431Sigor@sysoev.ru     nxt_req_conn_link_t     *rc;
553425Smax.romanov@nginx.com 
554425Smax.romanov@nginx.com     nxt_assert(task->thread->engine == ra->work.data);
555425Smax.romanov@nginx.com     nxt_assert(ra->use_count == 0);
556425Smax.romanov@nginx.com 
557343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
558343Smax.romanov@nginx.com 
559343Smax.romanov@nginx.com     rc = ra->rc;
560343Smax.romanov@nginx.com 
561343Smax.romanov@nginx.com     if (rc != NULL) {
562423Smax.romanov@nginx.com         if (nxt_slow_path(ra->err_code != 0)) {
563431Sigor@sysoev.ru             nxt_http_request_error(task, rc->ap->request, ra->err_code);
564423Smax.romanov@nginx.com 
565423Smax.romanov@nginx.com         } else {
566423Smax.romanov@nginx.com             rc->app_port = ra->app_port;
567423Smax.romanov@nginx.com             rc->msg_info = ra->msg_info;
568423Smax.romanov@nginx.com 
569425Smax.romanov@nginx.com             if (rc->app->timeout != 0) {
570431Sigor@sysoev.ru                 rc->ap->timer.handler = nxt_router_app_timeout;
571615Smax.romanov@nginx.com                 rc->ap->timer_data = rc;
572431Sigor@sysoev.ru                 nxt_timer_add(task->thread->engine, &rc->ap->timer,
573425Smax.romanov@nginx.com                               rc->app->timeout);
574425Smax.romanov@nginx.com             }
575425Smax.romanov@nginx.com 
576423Smax.romanov@nginx.com             ra->app_port = NULL;
577423Smax.romanov@nginx.com             ra->msg_info.buf = NULL;
578423Smax.romanov@nginx.com         }
579343Smax.romanov@nginx.com 
580343Smax.romanov@nginx.com         rc->ra = NULL;
581343Smax.romanov@nginx.com         ra->rc = NULL;
582343Smax.romanov@nginx.com     }
583343Smax.romanov@nginx.com 
584343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
585343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
586343Smax.romanov@nginx.com 
587343Smax.romanov@nginx.com         ra->app_port = NULL;
588167Smax.romanov@nginx.com     }
589167Smax.romanov@nginx.com 
590423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &ra->msg_info, ra->stream);
591423Smax.romanov@nginx.com 
592430Sigor@sysoev.ru     mp = ra->mem_pool;
593430Sigor@sysoev.ru 
594430Sigor@sysoev.ru     if (mp != NULL) {
595430Sigor@sysoev.ru         nxt_mp_free(mp, ra);
596430Sigor@sysoev.ru         nxt_mp_release(mp);
597351Smax.romanov@nginx.com     }
598167Smax.romanov@nginx.com }
599167Smax.romanov@nginx.com 
600167Smax.romanov@nginx.com 
601425Smax.romanov@nginx.com static void
602425Smax.romanov@nginx.com nxt_router_ra_release_handler(nxt_task_t *task, void *obj, void *data)
603425Smax.romanov@nginx.com {
604425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
605425Smax.romanov@nginx.com 
606425Smax.romanov@nginx.com     ra = obj;
607425Smax.romanov@nginx.com 
608425Smax.romanov@nginx.com     nxt_assert(ra->work.data == data);
609425Smax.romanov@nginx.com 
610425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, -1);
611425Smax.romanov@nginx.com 
612425Smax.romanov@nginx.com     nxt_router_ra_release(task, ra);
613425Smax.romanov@nginx.com }
614425Smax.romanov@nginx.com 
615425Smax.romanov@nginx.com 
616425Smax.romanov@nginx.com static void
617425Smax.romanov@nginx.com nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i)
618425Smax.romanov@nginx.com {
619425Smax.romanov@nginx.com     int                 c;
620425Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
621425Smax.romanov@nginx.com 
622425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, i);
623425Smax.romanov@nginx.com 
624425Smax.romanov@nginx.com     if (i < 0 && c == -i) {
625425Smax.romanov@nginx.com         engine = ra->work.data;
626425Smax.romanov@nginx.com 
627425Smax.romanov@nginx.com         if (task->thread->engine == engine) {
628425Smax.romanov@nginx.com             nxt_router_ra_release(task, ra);
629425Smax.romanov@nginx.com 
630425Smax.romanov@nginx.com             return;
631425Smax.romanov@nginx.com         }
632425Smax.romanov@nginx.com 
633425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
634425Smax.romanov@nginx.com 
635425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release_handler;
636425Smax.romanov@nginx.com         ra->work.task = &engine->task;
637425Smax.romanov@nginx.com         ra->work.next = NULL;
638425Smax.romanov@nginx.com 
639425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
640425Smax.romanov@nginx.com                   ra->stream, engine);
641425Smax.romanov@nginx.com 
642425Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
643425Smax.romanov@nginx.com     }
644425Smax.romanov@nginx.com }
645425Smax.romanov@nginx.com 
646425Smax.romanov@nginx.com 
647423Smax.romanov@nginx.com nxt_inline void
648521Szelenkov@nginx.com nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char *str)
649345Smax.romanov@nginx.com {
650423Smax.romanov@nginx.com     ra->app_port = NULL;
651423Smax.romanov@nginx.com     ra->err_code = code;
652423Smax.romanov@nginx.com     ra->err_str = str;
653345Smax.romanov@nginx.com }
654345Smax.romanov@nginx.com 
655345Smax.romanov@nginx.com 
656427Smax.romanov@nginx.com nxt_inline void
657427Smax.romanov@nginx.com nxt_router_ra_pending(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
658427Smax.romanov@nginx.com {
659427Smax.romanov@nginx.com     nxt_queue_insert_tail(&ra->app_port->pending_requests,
660427Smax.romanov@nginx.com                           &ra->link_port_pending);
661427Smax.romanov@nginx.com     nxt_queue_insert_tail(&app->pending, &ra->link_app_pending);
662427Smax.romanov@nginx.com 
663427Smax.romanov@nginx.com     nxt_router_ra_inc_use(ra);
664427Smax.romanov@nginx.com 
665427Smax.romanov@nginx.com     ra->res_time = nxt_thread_monotonic_time(task->thread) + app->res_timeout;
666427Smax.romanov@nginx.com 
667427Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD enqueue to pending_requests", ra->stream);
668427Smax.romanov@nginx.com }
669427Smax.romanov@nginx.com 
670427Smax.romanov@nginx.com 
671425Smax.romanov@nginx.com nxt_inline nxt_bool_t
672425Smax.romanov@nginx.com nxt_queue_chk_remove(nxt_queue_link_t *lnk)
673425Smax.romanov@nginx.com {
674425Smax.romanov@nginx.com     if (lnk->next != NULL) {
675425Smax.romanov@nginx.com         nxt_queue_remove(lnk);
676425Smax.romanov@nginx.com 
677425Smax.romanov@nginx.com         lnk->next = NULL;
678425Smax.romanov@nginx.com 
679425Smax.romanov@nginx.com         return 1;
680425Smax.romanov@nginx.com     }
681425Smax.romanov@nginx.com 
682425Smax.romanov@nginx.com     return 0;
683425Smax.romanov@nginx.com }
684425Smax.romanov@nginx.com 
685425Smax.romanov@nginx.com 
686343Smax.romanov@nginx.com nxt_inline void
687343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
688343Smax.romanov@nginx.com {
689425Smax.romanov@nginx.com     int                 ra_use_delta;
690343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
691343Smax.romanov@nginx.com 
692343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
693343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
694343Smax.romanov@nginx.com 
695343Smax.romanov@nginx.com         rc->app_port = NULL;
696343Smax.romanov@nginx.com     }
697343Smax.romanov@nginx.com 
698423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &rc->msg_info, rc->stream);
699423Smax.romanov@nginx.com 
700343Smax.romanov@nginx.com     ra = rc->ra;
701343Smax.romanov@nginx.com 
702343Smax.romanov@nginx.com     if (ra != NULL) {
703343Smax.romanov@nginx.com         rc->ra = NULL;
704343Smax.romanov@nginx.com         ra->rc = NULL;
705343Smax.romanov@nginx.com 
706425Smax.romanov@nginx.com         ra_use_delta = 0;
707425Smax.romanov@nginx.com 
708343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
709343Smax.romanov@nginx.com 
710425Smax.romanov@nginx.com         if (ra->link_app_requests.next == NULL
711427Smax.romanov@nginx.com             && ra->link_port_pending.next == NULL
712427Smax.romanov@nginx.com             && ra->link_app_pending.next == NULL)
713425Smax.romanov@nginx.com         {
714425Smax.romanov@nginx.com             ra = NULL;
715343Smax.romanov@nginx.com 
716343Smax.romanov@nginx.com         } else {
717425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests);
718425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending);
719427Smax.romanov@nginx.com             nxt_queue_chk_remove(&ra->link_app_pending);
720343Smax.romanov@nginx.com         }
721343Smax.romanov@nginx.com 
722343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
723425Smax.romanov@nginx.com 
724425Smax.romanov@nginx.com         if (ra != NULL) {
725425Smax.romanov@nginx.com             nxt_router_ra_use(task, ra, ra_use_delta);
726425Smax.romanov@nginx.com         }
727343Smax.romanov@nginx.com     }
728343Smax.romanov@nginx.com 
729343Smax.romanov@nginx.com     if (rc->app != NULL) {
730343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
731343Smax.romanov@nginx.com 
732343Smax.romanov@nginx.com         rc->app = NULL;
733343Smax.romanov@nginx.com     }
734343Smax.romanov@nginx.com 
735346Smax.romanov@nginx.com     if (rc->ap != NULL) {
736615Smax.romanov@nginx.com         rc->ap->timer_data = NULL;
737615Smax.romanov@nginx.com 
738346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
739346Smax.romanov@nginx.com 
740346Smax.romanov@nginx.com         rc->ap = NULL;
741346Smax.romanov@nginx.com     }
742343Smax.romanov@nginx.com }
743343Smax.romanov@nginx.com 
744343Smax.romanov@nginx.com 
745141Smax.romanov@nginx.com void
746141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
747141Smax.romanov@nginx.com {
748141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
749141Smax.romanov@nginx.com 
750192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
751141Smax.romanov@nginx.com         return;
752141Smax.romanov@nginx.com     }
753141Smax.romanov@nginx.com 
754426Smax.romanov@nginx.com     if (msg->u.new_port == NULL
755426Smax.romanov@nginx.com         || msg->u.new_port->type != NXT_PROCESS_WORKER)
756347Smax.romanov@nginx.com     {
757192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
758141Smax.romanov@nginx.com     }
759192Smax.romanov@nginx.com 
760192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
761141Smax.romanov@nginx.com }
762141Smax.romanov@nginx.com 
763141Smax.romanov@nginx.com 
764139Sigor@sysoev.ru void
765139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
766115Sigor@sysoev.ru {
767198Sigor@sysoev.ru     nxt_int_t               ret;
768139Sigor@sysoev.ru     nxt_buf_t               *b;
769139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
770139Sigor@sysoev.ru 
771139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
772139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
773139Sigor@sysoev.ru         return;
77453Sigor@sysoev.ru     }
77553Sigor@sysoev.ru 
776494Spluknet@nginx.com     nxt_debug(task, "nxt_router_conf_data_handler(%O): %*s",
777423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf),
778493Spluknet@nginx.com               (size_t) nxt_buf_used_size(msg->buf), msg->buf->mem.pos);
779423Smax.romanov@nginx.com 
780591Sigor@sysoev.ru     tmcf->router_conf->router = nxt_router;
781139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
782139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
783198Sigor@sysoev.ru                                        msg->port_msg.pid,
784198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
785198Sigor@sysoev.ru 
786591Sigor@sysoev.ru     b = nxt_buf_chk_make_plain(tmcf->router_conf->mem_pool,
787591Sigor@sysoev.ru                                msg->buf, msg->size);
788551Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
789551Smax.romanov@nginx.com         nxt_router_conf_error(task, tmcf);
790551Smax.romanov@nginx.com 
791551Smax.romanov@nginx.com         return;
792551Smax.romanov@nginx.com     }
793551Smax.romanov@nginx.com 
794198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
795198Sigor@sysoev.ru 
796198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
797198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
798198Sigor@sysoev.ru 
799198Sigor@sysoev.ru     } else {
800198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
801139Sigor@sysoev.ru     }
802