xref: /unit/src/nxt_router.c (revision 431)
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>
10*431Sigor@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;
15318Smax.romanov@nginx.com     uint32_t          workers;
16318Smax.romanov@nginx.com     nxt_msec_t        timeout;
17427Smax.romanov@nginx.com     nxt_msec_t        res_timeout;
18318Smax.romanov@nginx.com     uint32_t          requests;
19318Smax.romanov@nginx.com     nxt_conf_value_t  *limits_value;
20133Sigor@sysoev.ru } nxt_router_app_conf_t;
21133Sigor@sysoev.ru 
22133Sigor@sysoev.ru 
23133Sigor@sysoev.ru typedef struct {
24133Sigor@sysoev.ru     nxt_str_t  application;
25115Sigor@sysoev.ru } nxt_router_listener_conf_t;
26115Sigor@sysoev.ru 
27115Sigor@sysoev.ru 
28423Smax.romanov@nginx.com typedef struct nxt_msg_info_s {
29423Smax.romanov@nginx.com     nxt_buf_t                 *buf;
30423Smax.romanov@nginx.com     nxt_port_mmap_tracking_t  tracking;
31423Smax.romanov@nginx.com     nxt_work_handler_t        completion_handler;
32423Smax.romanov@nginx.com } nxt_msg_info_t;
33423Smax.romanov@nginx.com 
34423Smax.romanov@nginx.com 
35167Smax.romanov@nginx.com typedef struct nxt_req_app_link_s nxt_req_app_link_t;
36141Smax.romanov@nginx.com 
37141Smax.romanov@nginx.com 
38318Smax.romanov@nginx.com typedef struct {
39*431Sigor@sysoev.ru     uint32_t                 stream;
40*431Sigor@sysoev.ru     nxt_app_t                *app;
41*431Sigor@sysoev.ru     nxt_port_t               *app_port;
42*431Sigor@sysoev.ru     nxt_app_parse_ctx_t      *ap;
43*431Sigor@sysoev.ru     nxt_msg_info_t           msg_info;
44*431Sigor@sysoev.ru     nxt_req_app_link_t       *ra;
45*431Sigor@sysoev.ru 
46*431Sigor@sysoev.ru     nxt_queue_link_t         link;     /* for nxt_conn_t.requests */
47318Smax.romanov@nginx.com } nxt_req_conn_link_t;
48318Smax.romanov@nginx.com 
49318Smax.romanov@nginx.com 
50167Smax.romanov@nginx.com struct nxt_req_app_link_s {
51318Smax.romanov@nginx.com     uint32_t             stream;
52425Smax.romanov@nginx.com     nxt_atomic_t         use_count;
53167Smax.romanov@nginx.com     nxt_port_t           *app_port;
54167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
55167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
56423Smax.romanov@nginx.com     nxt_msg_info_t       msg_info;
57167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
58167Smax.romanov@nginx.com 
59427Smax.romanov@nginx.com     nxt_nsec_t           res_time;
60427Smax.romanov@nginx.com 
61425Smax.romanov@nginx.com     nxt_queue_link_t     link_app_requests; /* for nxt_app_t.requests */
62425Smax.romanov@nginx.com     nxt_queue_link_t     link_port_pending; /* for nxt_port_t.pending_requests */
63427Smax.romanov@nginx.com     nxt_queue_link_t     link_app_pending;  /* for nxt_app_t.pending */
64167Smax.romanov@nginx.com 
65167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
66167Smax.romanov@nginx.com     nxt_work_t           work;
67345Smax.romanov@nginx.com 
68345Smax.romanov@nginx.com     int                  err_code;
69345Smax.romanov@nginx.com     const char           *err_str;
70167Smax.romanov@nginx.com };
71167Smax.romanov@nginx.com 
72167Smax.romanov@nginx.com 
73198Sigor@sysoev.ru typedef struct {
74198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
75198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
76198Sigor@sysoev.ru } nxt_socket_rpc_t;
77198Sigor@sysoev.ru 
78198Sigor@sysoev.ru 
79427Smax.romanov@nginx.com struct nxt_port_select_state_s {
80427Smax.romanov@nginx.com     nxt_app_t           *app;
81427Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
82427Smax.romanov@nginx.com 
83427Smax.romanov@nginx.com     nxt_port_t          *failed_port;
84427Smax.romanov@nginx.com     int                 failed_port_use_delta;
85427Smax.romanov@nginx.com 
86427Smax.romanov@nginx.com     nxt_bool_t          can_start_worker;
87427Smax.romanov@nginx.com     nxt_req_app_link_t  *shared_ra;
88427Smax.romanov@nginx.com     nxt_port_t          *port;
89427Smax.romanov@nginx.com };
90427Smax.romanov@nginx.com 
91427Smax.romanov@nginx.com typedef struct nxt_port_select_state_s nxt_port_select_state_t;
92427Smax.romanov@nginx.com 
93427Smax.romanov@nginx.com static void nxt_router_port_select(nxt_task_t *task,
94427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
95427Smax.romanov@nginx.com 
96427Smax.romanov@nginx.com static nxt_int_t nxt_router_port_post_select(nxt_task_t *task,
97427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
98427Smax.romanov@nginx.com 
99343Smax.romanov@nginx.com static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app);
100343Smax.romanov@nginx.com 
101425Smax.romanov@nginx.com nxt_inline void
102425Smax.romanov@nginx.com nxt_router_ra_inc_use(nxt_req_app_link_t *ra)
103425Smax.romanov@nginx.com {
104425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, 1);
105425Smax.romanov@nginx.com }
106425Smax.romanov@nginx.com 
107425Smax.romanov@nginx.com nxt_inline void
108425Smax.romanov@nginx.com nxt_router_ra_dec_use(nxt_req_app_link_t *ra)
109425Smax.romanov@nginx.com {
110425Smax.romanov@nginx.com     int  c;
111425Smax.romanov@nginx.com 
112425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, -1);
113425Smax.romanov@nginx.com 
114425Smax.romanov@nginx.com     nxt_assert(c > 1);
115425Smax.romanov@nginx.com }
116425Smax.romanov@nginx.com 
117425Smax.romanov@nginx.com static void nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i);
118425Smax.romanov@nginx.com 
119139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
120198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
121198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
122139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
123139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
124139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
125139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
126193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
12753Sigor@sysoev.ru 
128115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
129115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
130133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
131133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
132133Sigor@sysoev.ru     nxt_str_t *name);
133198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
134198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
135198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
136198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
137198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
138198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
139359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
140359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
141359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
142359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
14353Sigor@sysoev.ru 
14453Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
14553Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
14653Sigor@sysoev.ru     const nxt_event_interface_t *interface);
147115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
148115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
149115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
150115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
151115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
152115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
153154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
154154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
155154Sigor@sysoev.ru     nxt_work_handler_t handler);
156313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
157313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
158139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
159139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
16053Sigor@sysoev.ru 
16153Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
16253Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
16353Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
16453Sigor@sysoev.ru     nxt_event_engine_t *engine);
165343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
166133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
16753Sigor@sysoev.ru 
168315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
169315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
170315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
171315Sigor@sysoev.ru     nxt_work_t *jobs);
17253Sigor@sysoev.ru 
17353Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
17453Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
17553Sigor@sysoev.ru     void *data);
17653Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
17753Sigor@sysoev.ru     void *data);
17853Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
17953Sigor@sysoev.ru     void *data);
180313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
181313Sigor@sysoev.ru     void *data);
18253Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
18353Sigor@sysoev.ru     void *data);
18453Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
18553Sigor@sysoev.ru     void *data);
186359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
187359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
18853Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
18953Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
19053Sigor@sysoev.ru 
191343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
192343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
193343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
194343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
195343Smax.romanov@nginx.com 
196343Smax.romanov@nginx.com static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app);
197343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
198343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
199427Smax.romanov@nginx.com static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_app_t *app,
200427Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
201141Smax.romanov@nginx.com 
202425Smax.romanov@nginx.com static void nxt_router_app_prepare_request(nxt_task_t *task,
203343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
204216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
205216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
206216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
207216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
208216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
209216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
21053Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
211318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
212*431Sigor@sysoev.ru 
213*431Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_request_send_state;
214*431Sigor@sysoev.ru static void nxt_http_request_send_body(nxt_task_t *task, void *obj, void *data);
215141Smax.romanov@nginx.com 
216119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
21720Sigor@sysoev.ru 
218216Sigor@sysoev.ru 
219216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
220216Sigor@sysoev.ru     nxt_python_prepare_msg,
221216Sigor@sysoev.ru     nxt_php_prepare_msg,
222216Sigor@sysoev.ru     nxt_go_prepare_msg,
223216Sigor@sysoev.ru };
224216Sigor@sysoev.ru 
225216Sigor@sysoev.ru 
22620Sigor@sysoev.ru nxt_int_t
227141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
22820Sigor@sysoev.ru {
229141Smax.romanov@nginx.com     nxt_int_t      ret;
230141Smax.romanov@nginx.com     nxt_router_t   *router;
231141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
232141Smax.romanov@nginx.com 
233141Smax.romanov@nginx.com     rt = task->thread->runtime;
23453Sigor@sysoev.ru 
235*431Sigor@sysoev.ru     ret = nxt_http_init(task, rt);
23688Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
23788Smax.romanov@nginx.com         return ret;
23888Smax.romanov@nginx.com     }
23988Smax.romanov@nginx.com 
24053Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
24153Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
24253Sigor@sysoev.ru         return NXT_ERROR;
24353Sigor@sysoev.ru     }
24453Sigor@sysoev.ru 
24553Sigor@sysoev.ru     nxt_queue_init(&router->engines);
24653Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
247133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
24853Sigor@sysoev.ru 
249119Smax.romanov@nginx.com     nxt_router = router;
250119Smax.romanov@nginx.com 
251115Sigor@sysoev.ru     return NXT_OK;
252115Sigor@sysoev.ru }
253115Sigor@sysoev.ru 
254115Sigor@sysoev.ru 
255343Smax.romanov@nginx.com static void
256343Smax.romanov@nginx.com nxt_router_start_worker_handler(nxt_task_t *task, nxt_port_t *port, void *data)
257167Smax.romanov@nginx.com {
258343Smax.romanov@nginx.com     size_t         size;
259343Smax.romanov@nginx.com     uint32_t       stream;
260430Sigor@sysoev.ru     nxt_mp_t       *mp;
261343Smax.romanov@nginx.com     nxt_app_t      *app;
262343Smax.romanov@nginx.com     nxt_buf_t      *b;
263343Smax.romanov@nginx.com     nxt_port_t     *main_port;
264343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
265343Smax.romanov@nginx.com 
266343Smax.romanov@nginx.com     app = data;
267167Smax.romanov@nginx.com 
268167Smax.romanov@nginx.com     rt = task->thread->runtime;
269240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
270167Smax.romanov@nginx.com 
271343Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start worker", &app->name, app);
272343Smax.romanov@nginx.com 
273343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
274343Smax.romanov@nginx.com 
275343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
276343Smax.romanov@nginx.com 
277343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
278343Smax.romanov@nginx.com         goto failed;
279167Smax.romanov@nginx.com     }
280167Smax.romanov@nginx.com 
281343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
282343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
283343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
284343Smax.romanov@nginx.com 
285343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
286343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
287343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
288343Smax.romanov@nginx.com                                            -1, app);
289343Smax.romanov@nginx.com 
290343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
291430Sigor@sysoev.ru         mp = b->data;
292430Sigor@sysoev.ru         nxt_mp_free(mp, b);
293430Sigor@sysoev.ru         nxt_mp_release(mp);
294343Smax.romanov@nginx.com 
295343Smax.romanov@nginx.com         goto failed;
296343Smax.romanov@nginx.com     }
297343Smax.romanov@nginx.com 
298343Smax.romanov@nginx.com     nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
299343Smax.romanov@nginx.com                           stream, port->id, b);
300343Smax.romanov@nginx.com 
301343Smax.romanov@nginx.com     return;
302343Smax.romanov@nginx.com 
303343Smax.romanov@nginx.com failed:
304343Smax.romanov@nginx.com 
305343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
306343Smax.romanov@nginx.com 
307343Smax.romanov@nginx.com     app->pending_workers--;
308343Smax.romanov@nginx.com 
309343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
310343Smax.romanov@nginx.com 
311343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
312167Smax.romanov@nginx.com }
313167Smax.romanov@nginx.com 
314167Smax.romanov@nginx.com 
315343Smax.romanov@nginx.com static nxt_int_t
316343Smax.romanov@nginx.com nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app)
317141Smax.romanov@nginx.com {
318343Smax.romanov@nginx.com     nxt_int_t      res;
319343Smax.romanov@nginx.com     nxt_port_t     *router_port;
320343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
321343Smax.romanov@nginx.com 
322343Smax.romanov@nginx.com     rt = task->thread->runtime;
323343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
324343Smax.romanov@nginx.com 
325343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
326343Smax.romanov@nginx.com 
327343Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_worker_handler,
328343Smax.romanov@nginx.com                         app);
329343Smax.romanov@nginx.com 
330343Smax.romanov@nginx.com     if (res == NXT_OK) {
331343Smax.romanov@nginx.com         return res;
332318Smax.romanov@nginx.com     }
333318Smax.romanov@nginx.com 
334343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
335343Smax.romanov@nginx.com 
336343Smax.romanov@nginx.com     app->pending_workers--;
337343Smax.romanov@nginx.com 
338343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
339343Smax.romanov@nginx.com 
340343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
341343Smax.romanov@nginx.com 
342343Smax.romanov@nginx.com     return NXT_ERROR;
343318Smax.romanov@nginx.com }
344318Smax.romanov@nginx.com 
345318Smax.romanov@nginx.com 
346351Smax.romanov@nginx.com nxt_inline void
347351Smax.romanov@nginx.com nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra,
348351Smax.romanov@nginx.com     nxt_req_conn_link_t *rc)
349167Smax.romanov@nginx.com {
350318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
351351Smax.romanov@nginx.com 
352318Smax.romanov@nginx.com     engine = task->thread->engine;
353167Smax.romanov@nginx.com 
354167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
355167Smax.romanov@nginx.com 
356318Smax.romanov@nginx.com     ra->stream = rc->stream;
357425Smax.romanov@nginx.com     ra->use_count = 1;
358167Smax.romanov@nginx.com     ra->rc = rc;
359318Smax.romanov@nginx.com     rc->ra = ra;
360318Smax.romanov@nginx.com     ra->reply_port = engine->port;
361351Smax.romanov@nginx.com     ra->ap = rc->ap;
362167Smax.romanov@nginx.com 
363167Smax.romanov@nginx.com     ra->work.handler = NULL;
364318Smax.romanov@nginx.com     ra->work.task = &engine->task;
365167Smax.romanov@nginx.com     ra->work.obj = ra;
366318Smax.romanov@nginx.com     ra->work.data = engine;
367351Smax.romanov@nginx.com }
368351Smax.romanov@nginx.com 
369351Smax.romanov@nginx.com 
370351Smax.romanov@nginx.com nxt_inline nxt_req_app_link_t *
371351Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src)
372351Smax.romanov@nginx.com {
373351Smax.romanov@nginx.com     nxt_mp_t            *mp;
374351Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
375351Smax.romanov@nginx.com 
376425Smax.romanov@nginx.com     if (ra_src->mem_pool != NULL) {
377425Smax.romanov@nginx.com         return ra_src;
378425Smax.romanov@nginx.com     }
379425Smax.romanov@nginx.com 
380351Smax.romanov@nginx.com     mp = ra_src->ap->mem_pool;
381351Smax.romanov@nginx.com 
382430Sigor@sysoev.ru     ra = nxt_mp_alloc(mp, sizeof(nxt_req_app_link_t));
383351Smax.romanov@nginx.com 
384351Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
385351Smax.romanov@nginx.com 
386351Smax.romanov@nginx.com         ra_src->rc->ra = NULL;
387351Smax.romanov@nginx.com         ra_src->rc = NULL;
388351Smax.romanov@nginx.com 
389351Smax.romanov@nginx.com         return NULL;
390351Smax.romanov@nginx.com     }
391351Smax.romanov@nginx.com 
392430Sigor@sysoev.ru     nxt_mp_retain(mp);
393430Sigor@sysoev.ru 
394351Smax.romanov@nginx.com     nxt_router_ra_init(task, ra, ra_src->rc);
395351Smax.romanov@nginx.com 
396351Smax.romanov@nginx.com     ra->mem_pool = mp;
397167Smax.romanov@nginx.com 
398167Smax.romanov@nginx.com     return ra;
399167Smax.romanov@nginx.com }
400167Smax.romanov@nginx.com 
401167Smax.romanov@nginx.com 
402423Smax.romanov@nginx.com nxt_inline nxt_bool_t
403423Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info,
404423Smax.romanov@nginx.com     uint32_t stream)
405423Smax.romanov@nginx.com {
406423Smax.romanov@nginx.com     nxt_buf_t   *b, *next;
407423Smax.romanov@nginx.com     nxt_bool_t  cancelled;
408423Smax.romanov@nginx.com 
409423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
410423Smax.romanov@nginx.com         return 0;
411423Smax.romanov@nginx.com     }
412423Smax.romanov@nginx.com 
413423Smax.romanov@nginx.com     cancelled = nxt_port_mmap_tracking_cancel(task, &msg_info->tracking,
414423Smax.romanov@nginx.com                                               stream);
415423Smax.romanov@nginx.com 
416423Smax.romanov@nginx.com     if (cancelled) {
417423Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router", stream);
418423Smax.romanov@nginx.com     }
419423Smax.romanov@nginx.com 
420423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
421423Smax.romanov@nginx.com         next = b->next;
422423Smax.romanov@nginx.com 
423423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
424423Smax.romanov@nginx.com 
425423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
426423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
427423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
428423Smax.romanov@nginx.com         }
429423Smax.romanov@nginx.com     }
430423Smax.romanov@nginx.com 
431423Smax.romanov@nginx.com     msg_info->buf = NULL;
432423Smax.romanov@nginx.com 
433423Smax.romanov@nginx.com     return cancelled;
434423Smax.romanov@nginx.com }
435423Smax.romanov@nginx.com 
436423Smax.romanov@nginx.com 
437167Smax.romanov@nginx.com static void
438425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra);
439425Smax.romanov@nginx.com 
440425Smax.romanov@nginx.com 
441425Smax.romanov@nginx.com static void
442425Smax.romanov@nginx.com nxt_router_ra_update_peer_handler(nxt_task_t *task, void *obj, void *data)
443167Smax.romanov@nginx.com {
444425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
445425Smax.romanov@nginx.com 
446425Smax.romanov@nginx.com     ra = obj;
447425Smax.romanov@nginx.com 
448425Smax.romanov@nginx.com     nxt_router_ra_update_peer(task, ra);
449425Smax.romanov@nginx.com 
450425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
451425Smax.romanov@nginx.com }
452425Smax.romanov@nginx.com 
453425Smax.romanov@nginx.com 
454425Smax.romanov@nginx.com static void
455425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra)
456425Smax.romanov@nginx.com {
457343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
458343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
459318Smax.romanov@nginx.com 
460425Smax.romanov@nginx.com     engine = ra->work.data;
461318Smax.romanov@nginx.com 
462343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
463425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
464425Smax.romanov@nginx.com 
465425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_update_peer_handler;
466318Smax.romanov@nginx.com         ra->work.task = &engine->task;
467318Smax.romanov@nginx.com         ra->work.next = NULL;
468318Smax.romanov@nginx.com 
469425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post update peer to %p",
470318Smax.romanov@nginx.com                   ra->stream, engine);
471318Smax.romanov@nginx.com 
472318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
473318Smax.romanov@nginx.com 
474318Smax.romanov@nginx.com         return;
475318Smax.romanov@nginx.com     }
476318Smax.romanov@nginx.com 
477425Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD update peer", ra->stream);
478425Smax.romanov@nginx.com 
479425Smax.romanov@nginx.com     rc = ra->rc;
480425Smax.romanov@nginx.com 
481425Smax.romanov@nginx.com     if (rc != NULL && ra->app_port != NULL) {
482425Smax.romanov@nginx.com         nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_port->pid);
483425Smax.romanov@nginx.com     }
484425Smax.romanov@nginx.com 
485425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
486425Smax.romanov@nginx.com }
487425Smax.romanov@nginx.com 
488425Smax.romanov@nginx.com 
489425Smax.romanov@nginx.com static void
490425Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, nxt_req_app_link_t *ra)
491425Smax.romanov@nginx.com {
492*431Sigor@sysoev.ru     nxt_mp_t                *mp;
493*431Sigor@sysoev.ru     nxt_req_conn_link_t     *rc;
494425Smax.romanov@nginx.com 
495425Smax.romanov@nginx.com     nxt_assert(task->thread->engine == ra->work.data);
496425Smax.romanov@nginx.com     nxt_assert(ra->use_count == 0);
497425Smax.romanov@nginx.com 
498343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
499343Smax.romanov@nginx.com 
500343Smax.romanov@nginx.com     rc = ra->rc;
501343Smax.romanov@nginx.com 
502343Smax.romanov@nginx.com     if (rc != NULL) {
503423Smax.romanov@nginx.com         if (nxt_slow_path(ra->err_code != 0)) {
504*431Sigor@sysoev.ru             nxt_http_request_error(task, rc->ap->request, ra->err_code);
505423Smax.romanov@nginx.com 
506423Smax.romanov@nginx.com         } else {
507423Smax.romanov@nginx.com             rc->app_port = ra->app_port;
508423Smax.romanov@nginx.com             rc->msg_info = ra->msg_info;
509423Smax.romanov@nginx.com 
510425Smax.romanov@nginx.com             if (rc->app->timeout != 0) {
511*431Sigor@sysoev.ru                 rc->ap->timer.handler = nxt_router_app_timeout;
512*431Sigor@sysoev.ru                 nxt_timer_add(task->thread->engine, &rc->ap->timer,
513425Smax.romanov@nginx.com                               rc->app->timeout);
514425Smax.romanov@nginx.com             }
515425Smax.romanov@nginx.com 
516423Smax.romanov@nginx.com             ra->app_port = NULL;
517423Smax.romanov@nginx.com             ra->msg_info.buf = NULL;
518423Smax.romanov@nginx.com         }
519343Smax.romanov@nginx.com 
520343Smax.romanov@nginx.com         rc->ra = NULL;
521343Smax.romanov@nginx.com         ra->rc = NULL;
522343Smax.romanov@nginx.com     }
523343Smax.romanov@nginx.com 
524343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
525343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
526343Smax.romanov@nginx.com 
527343Smax.romanov@nginx.com         ra->app_port = NULL;
528167Smax.romanov@nginx.com     }
529167Smax.romanov@nginx.com 
530423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &ra->msg_info, ra->stream);
531423Smax.romanov@nginx.com 
532430Sigor@sysoev.ru     mp = ra->mem_pool;
533430Sigor@sysoev.ru 
534430Sigor@sysoev.ru     if (mp != NULL) {
535430Sigor@sysoev.ru         nxt_mp_free(mp, ra);
536430Sigor@sysoev.ru         nxt_mp_release(mp);
537351Smax.romanov@nginx.com     }
538167Smax.romanov@nginx.com }
539167Smax.romanov@nginx.com 
540167Smax.romanov@nginx.com 
541425Smax.romanov@nginx.com static void
542425Smax.romanov@nginx.com nxt_router_ra_release_handler(nxt_task_t *task, void *obj, void *data)
543425Smax.romanov@nginx.com {
544425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
545425Smax.romanov@nginx.com 
546425Smax.romanov@nginx.com     ra = obj;
547425Smax.romanov@nginx.com 
548425Smax.romanov@nginx.com     nxt_assert(ra->work.data == data);
549425Smax.romanov@nginx.com 
550425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, -1);
551425Smax.romanov@nginx.com 
552425Smax.romanov@nginx.com     nxt_router_ra_release(task, ra);
553425Smax.romanov@nginx.com }
554425Smax.romanov@nginx.com 
555425Smax.romanov@nginx.com 
556425Smax.romanov@nginx.com static void
557425Smax.romanov@nginx.com nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i)
558425Smax.romanov@nginx.com {
559425Smax.romanov@nginx.com     int                 c;
560425Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
561425Smax.romanov@nginx.com 
562425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, i);
563425Smax.romanov@nginx.com 
564425Smax.romanov@nginx.com     if (i < 0 && c == -i) {
565425Smax.romanov@nginx.com         engine = ra->work.data;
566425Smax.romanov@nginx.com 
567425Smax.romanov@nginx.com         if (task->thread->engine == engine) {
568425Smax.romanov@nginx.com             nxt_router_ra_release(task, ra);
569425Smax.romanov@nginx.com 
570425Smax.romanov@nginx.com             return;
571425Smax.romanov@nginx.com         }
572425Smax.romanov@nginx.com 
573425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
574425Smax.romanov@nginx.com 
575425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release_handler;
576425Smax.romanov@nginx.com         ra->work.task = &engine->task;
577425Smax.romanov@nginx.com         ra->work.next = NULL;
578425Smax.romanov@nginx.com 
579425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
580425Smax.romanov@nginx.com                   ra->stream, engine);
581425Smax.romanov@nginx.com 
582425Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
583425Smax.romanov@nginx.com     }
584425Smax.romanov@nginx.com }
585425Smax.romanov@nginx.com 
586425Smax.romanov@nginx.com 
587423Smax.romanov@nginx.com nxt_inline void
588423Smax.romanov@nginx.com nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str)
589345Smax.romanov@nginx.com {
590423Smax.romanov@nginx.com     ra->app_port = NULL;
591423Smax.romanov@nginx.com     ra->err_code = code;
592423Smax.romanov@nginx.com     ra->err_str = str;
593345Smax.romanov@nginx.com }
594345Smax.romanov@nginx.com 
595345Smax.romanov@nginx.com 
596427Smax.romanov@nginx.com nxt_inline void
597427Smax.romanov@nginx.com nxt_router_ra_pending(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
598427Smax.romanov@nginx.com {
599427Smax.romanov@nginx.com     nxt_queue_insert_tail(&ra->app_port->pending_requests,
600427Smax.romanov@nginx.com                           &ra->link_port_pending);
601427Smax.romanov@nginx.com     nxt_queue_insert_tail(&app->pending, &ra->link_app_pending);
602427Smax.romanov@nginx.com 
603427Smax.romanov@nginx.com     nxt_router_ra_inc_use(ra);
604427Smax.romanov@nginx.com 
605427Smax.romanov@nginx.com     ra->res_time = nxt_thread_monotonic_time(task->thread) + app->res_timeout;
606427Smax.romanov@nginx.com 
607427Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD enqueue to pending_requests", ra->stream);
608427Smax.romanov@nginx.com }
609427Smax.romanov@nginx.com 
610427Smax.romanov@nginx.com 
611425Smax.romanov@nginx.com nxt_inline nxt_bool_t
612425Smax.romanov@nginx.com nxt_queue_chk_remove(nxt_queue_link_t *lnk)
613425Smax.romanov@nginx.com {
614425Smax.romanov@nginx.com     if (lnk->next != NULL) {
615425Smax.romanov@nginx.com         nxt_queue_remove(lnk);
616425Smax.romanov@nginx.com 
617425Smax.romanov@nginx.com         lnk->next = NULL;
618425Smax.romanov@nginx.com 
619425Smax.romanov@nginx.com         return 1;
620425Smax.romanov@nginx.com     }
621425Smax.romanov@nginx.com 
622425Smax.romanov@nginx.com     return 0;
623425Smax.romanov@nginx.com }
624425Smax.romanov@nginx.com 
625425Smax.romanov@nginx.com 
626343Smax.romanov@nginx.com nxt_inline void
627343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
628343Smax.romanov@nginx.com {
629425Smax.romanov@nginx.com     int                 ra_use_delta;
630343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
631343Smax.romanov@nginx.com 
632343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
633343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
634343Smax.romanov@nginx.com 
635343Smax.romanov@nginx.com         rc->app_port = NULL;
636343Smax.romanov@nginx.com     }
637343Smax.romanov@nginx.com 
638423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &rc->msg_info, rc->stream);
639423Smax.romanov@nginx.com 
640343Smax.romanov@nginx.com     ra = rc->ra;
641343Smax.romanov@nginx.com 
642343Smax.romanov@nginx.com     if (ra != NULL) {
643343Smax.romanov@nginx.com         rc->ra = NULL;
644343Smax.romanov@nginx.com         ra->rc = NULL;
645343Smax.romanov@nginx.com 
646425Smax.romanov@nginx.com         ra_use_delta = 0;
647425Smax.romanov@nginx.com 
648343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
649343Smax.romanov@nginx.com 
650425Smax.romanov@nginx.com         if (ra->link_app_requests.next == NULL
651427Smax.romanov@nginx.com             && ra->link_port_pending.next == NULL
652427Smax.romanov@nginx.com             && ra->link_app_pending.next == NULL)
653425Smax.romanov@nginx.com         {
654425Smax.romanov@nginx.com             ra = NULL;
655343Smax.romanov@nginx.com 
656343Smax.romanov@nginx.com         } else {
657425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests);
658425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending);
659427Smax.romanov@nginx.com             nxt_queue_chk_remove(&ra->link_app_pending);
660343Smax.romanov@nginx.com         }
661343Smax.romanov@nginx.com 
662343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
663425Smax.romanov@nginx.com 
664425Smax.romanov@nginx.com         if (ra != NULL) {
665425Smax.romanov@nginx.com             nxt_router_ra_use(task, ra, ra_use_delta);
666425Smax.romanov@nginx.com         }
667343Smax.romanov@nginx.com     }
668343Smax.romanov@nginx.com 
669343Smax.romanov@nginx.com     if (rc->app != NULL) {
670343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
671343Smax.romanov@nginx.com 
672343Smax.romanov@nginx.com         rc->app = NULL;
673343Smax.romanov@nginx.com     }
674343Smax.romanov@nginx.com 
675346Smax.romanov@nginx.com     if (rc->ap != NULL) {
676346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
677346Smax.romanov@nginx.com 
678346Smax.romanov@nginx.com         rc->ap = NULL;
679346Smax.romanov@nginx.com     }
680343Smax.romanov@nginx.com }
681343Smax.romanov@nginx.com 
682343Smax.romanov@nginx.com 
683141Smax.romanov@nginx.com void
684141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
685141Smax.romanov@nginx.com {
686141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
687141Smax.romanov@nginx.com 
688192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
689141Smax.romanov@nginx.com         return;
690141Smax.romanov@nginx.com     }
691141Smax.romanov@nginx.com 
692426Smax.romanov@nginx.com     if (msg->u.new_port == NULL
693426Smax.romanov@nginx.com         || msg->u.new_port->type != NXT_PROCESS_WORKER)
694347Smax.romanov@nginx.com     {
695192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
696141Smax.romanov@nginx.com     }
697192Smax.romanov@nginx.com 
698192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
699141Smax.romanov@nginx.com }
700141Smax.romanov@nginx.com 
701141Smax.romanov@nginx.com 
702139Sigor@sysoev.ru void
703139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
704115Sigor@sysoev.ru {
705198Sigor@sysoev.ru     nxt_int_t               ret;
706139Sigor@sysoev.ru     nxt_buf_t               *b;
707139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
708139Sigor@sysoev.ru 
709139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
710139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
711139Sigor@sysoev.ru         return;
71253Sigor@sysoev.ru     }
71353Sigor@sysoev.ru 
714423Smax.romanov@nginx.com     nxt_debug(task, "nxt_router_conf_data_handler(%d): %*s",
715423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf),
716423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf), msg->buf->mem.pos);
717423Smax.romanov@nginx.com 
718352Smax.romanov@nginx.com     b = nxt_buf_chk_make_plain(tmcf->conf->mem_pool, msg->buf, msg->size);
719352Smax.romanov@nginx.com 
720352Smax.romanov@nginx.com     nxt_assert(b != NULL);
721352Smax.romanov@nginx.com 
722139Sigor@sysoev.ru     tmcf->conf->router = nxt_router;
723139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
724139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
725198Sigor@sysoev.ru                                        msg->port_msg.pid,
726198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
727198Sigor@sysoev.ru 
728198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
729198Sigor@sysoev.ru 
730198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
731198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
732198Sigor@sysoev.ru 
733198Sigor@sysoev.ru     } else {
734198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
735139Sigor@sysoev.ru     }
73653Sigor@sysoev.ru }
73753Sigor@sysoev.ru 
73853Sigor@sysoev.ru 
739347Smax.romanov@nginx.com static void
740347Smax.romanov@nginx.com nxt_router_worker_remove_pid(nxt_task_t *task, nxt_port_t *port, void *data)
741347Smax.romanov@nginx.com {
742347Smax.romanov@nginx.com     union {
743347Smax.romanov@nginx.com         nxt_pid_t  removed_pid;
744347Smax.romanov@nginx.com         void       *data;
745347Smax.romanov@nginx.com     } u;
746347Smax.romanov@nginx.com 
747347Smax.romanov@nginx.com     u.data = data;
748347Smax.romanov@nginx.com 
749347Smax.romanov@nginx.com     nxt_port_rpc_remove_peer(task, port, u.removed_pid);
750347Smax.romanov@nginx.com }
751347Smax.romanov@nginx.com 
752347Smax.romanov@nginx.com 
753192Smax.romanov@nginx.com void
754192Smax.romanov@nginx.com nxt_router_remove_pid_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
755192Smax.romanov@nginx.com {
756347Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
757318Smax.romanov@nginx.com 
758192Smax.romanov@nginx.com     nxt_port_remove_pid_handler(task, msg);
759192Smax.romanov@nginx.com 
760192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
761192Smax.romanov@nginx.com         return;
762192Smax.romanov@nginx.com     }
763192Smax.romanov@nginx.com 
764318Smax.romanov@nginx.com     nxt_queue_each(engine, &nxt_router->engines, nxt_event_engine_t, link0)
765318Smax.romanov@nginx.com     {
766347Smax.romanov@nginx.com         nxt_port_post(task, engine->port, nxt_router_worker_remove_pid,
767347Smax.romanov@nginx.com                       msg->u.data);
768318Smax.romanov@nginx.com     }
769318Smax.romanov@nginx.com     nxt_queue_loop;
770318Smax.romanov@nginx.com 
771192Smax.romanov@nginx.com     msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
772192Smax.romanov@nginx.com 
773192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
774192Smax.romanov@nginx.com }
775192Smax.romanov@nginx.com 
776192Smax.romanov@nginx.com 
77753Sigor@sysoev.ru static nxt_router_temp_conf_t *
778139Sigor@sysoev.ru nxt_router_temp_conf(nxt_task_t *task)
77953Sigor@sysoev.ru {
78065Sigor@sysoev.ru     nxt_mp_t                *mp, *tmp;
78153Sigor@sysoev.ru     nxt_router_conf_t       *rtcf;
78253Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
78353Sigor@sysoev.ru 
78465Sigor@sysoev.ru     mp = nxt_mp_create(1024, 128, 256, 32);
78553Sigor@sysoev.ru     if (nxt_slow_path(mp == NULL)) {
78653Sigor@sysoev.ru         return NULL;
78753Sigor@sysoev.ru     }
78853Sigor@sysoev.ru 
78965Sigor@sysoev.ru     rtcf = nxt_mp_zget(mp, sizeof(nxt_router_conf_t));
79053Sigor@sysoev.ru     if (nxt_slow_path(rtcf == NULL)) {
79153Sigor@sysoev.ru         goto fail;
79253Sigor@sysoev.ru     }
79353Sigor@sysoev.ru 
79453Sigor@sysoev.ru     rtcf->mem_pool = mp;
79553Sigor@sysoev.ru 
79665Sigor@sysoev.ru     tmp = nxt_mp_create(1024, 128, 256, 32);
79753Sigor@sysoev.ru     if (nxt_slow_path(tmp == NULL)) {
79853Sigor@sysoev.ru         goto fail;
79953Sigor@sysoev.ru     }
80053Sigor@sysoev.ru 
80165Sigor@sysoev.ru     tmcf = nxt_mp_zget(tmp, sizeof(nxt_router_temp_conf_t));
80253Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
80353Sigor@sysoev.ru         goto temp_fail;
80453Sigor@sysoev.ru     }
80553Sigor@sysoev.ru 
80653Sigor@sysoev.ru     tmcf->mem_pool = tmp;
80753Sigor@sysoev.ru     tmcf->conf = rtcf;
808139Sigor@sysoev.ru     tmcf->count = 1;
809139Sigor@sysoev.ru     tmcf->engine = task->thread->engine;
81053Sigor@sysoev.ru 
81153Sigor@sysoev.ru     tmcf->engines = nxt_array_create(tmcf->mem_pool, 4,
81253Sigor@sysoev.ru                                      sizeof(nxt_router_engine_conf_t));
81353Sigor@sysoev.ru     if (nxt_slow_path(tmcf->engines == NULL)) {
81453Sigor@sysoev.ru         goto temp_fail;
81553Sigor@sysoev.ru     }
81653Sigor@sysoev.ru 
817