xref: /unit/src/nxt_router.c (revision 507)
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;
15*507Smax.romanov@nginx.com     uint32_t          processes;
16*507Smax.romanov@nginx.com     uint32_t          max_processes;
17*507Smax.romanov@nginx.com     uint32_t          spare_processes;
18318Smax.romanov@nginx.com     nxt_msec_t        timeout;
19427Smax.romanov@nginx.com     nxt_msec_t        res_timeout;
20*507Smax.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;
23*507Smax.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 
83*507Smax.romanov@nginx.com typedef struct {
84*507Smax.romanov@nginx.com     nxt_app_t               *app;
85*507Smax.romanov@nginx.com     nxt_router_temp_conf_t  *temp_conf;
86*507Smax.romanov@nginx.com } nxt_app_rpc_t;
87*507Smax.romanov@nginx.com 
88*507Smax.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 
96*507Smax.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 
109*507Smax.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 {
120425Smax.romanov@nginx.com     int  c;
121425Smax.romanov@nginx.com 
122425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, -1);
123425Smax.romanov@nginx.com 
124425Smax.romanov@nginx.com     nxt_assert(c > 1);
125425Smax.romanov@nginx.com }
126425Smax.romanov@nginx.com 
127425Smax.romanov@nginx.com static void nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i);
128425Smax.romanov@nginx.com 
129139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
130198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
131198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
132139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
133139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
134139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
135139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
136193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
13753Sigor@sysoev.ru 
138115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
139115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
140133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
141133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
142133Sigor@sysoev.ru     nxt_str_t *name);
143198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
144198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
145198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
146198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
147198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
148198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
149*507Smax.romanov@nginx.com static void nxt_router_app_rpc_create(nxt_task_t *task,
150*507Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_app_t *app);
151*507Smax.romanov@nginx.com static void nxt_router_app_prefork_ready(nxt_task_t *task,
152*507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
153*507Smax.romanov@nginx.com static void nxt_router_app_prefork_error(nxt_task_t *task,
154*507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
155359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
156359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
157359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
158359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
15953Sigor@sysoev.ru 
16053Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
16153Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
16253Sigor@sysoev.ru     const nxt_event_interface_t *interface);
163115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
164115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
165115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
166115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
167115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
168115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
169154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
170154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
171154Sigor@sysoev.ru     nxt_work_handler_t handler);
172313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
173313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
174139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
175139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
17653Sigor@sysoev.ru 
17753Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
17853Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
17953Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
18053Sigor@sysoev.ru     nxt_event_engine_t *engine);
181343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
182133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
18353Sigor@sysoev.ru 
184315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
185315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
186315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
187315Sigor@sysoev.ru     nxt_work_t *jobs);
18853Sigor@sysoev.ru 
18953Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
19053Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
19153Sigor@sysoev.ru     void *data);
19253Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
19353Sigor@sysoev.ru     void *data);
19453Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
19553Sigor@sysoev.ru     void *data);
196313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
197313Sigor@sysoev.ru     void *data);
19853Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
19953Sigor@sysoev.ru     void *data);
20053Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
20153Sigor@sysoev.ru     void *data);
202359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
203359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
20453Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
20553Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
20653Sigor@sysoev.ru 
207343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
208343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
209343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
210343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
211343Smax.romanov@nginx.com 
212*507Smax.romanov@nginx.com static void nxt_router_app_quit(nxt_task_t *task, nxt_app_t *app);
213343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
214343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
215427Smax.romanov@nginx.com static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_app_t *app,
216427Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
217141Smax.romanov@nginx.com 
218425Smax.romanov@nginx.com static void nxt_router_app_prepare_request(nxt_task_t *task,
219343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
220216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
221216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
222216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
223216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
224216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
225216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
22653Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
227318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
228*507Smax.romanov@nginx.com static void nxt_router_adjust_idle_timer(nxt_task_t *task, void *obj,
229*507Smax.romanov@nginx.com     void *data);
230*507Smax.romanov@nginx.com static void nxt_router_app_idle_timeout(nxt_task_t *task, void *obj,
231*507Smax.romanov@nginx.com     void *data);
232*507Smax.romanov@nginx.com static void nxt_router_app_release_handler(nxt_task_t *task, void *obj,
233*507Smax.romanov@nginx.com     void *data);
234431Sigor@sysoev.ru 
235431Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_request_send_state;
236431Sigor@sysoev.ru static void nxt_http_request_send_body(nxt_task_t *task, void *obj, void *data);
237141Smax.romanov@nginx.com 
238119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
23920Sigor@sysoev.ru 
240216Sigor@sysoev.ru 
241216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
242216Sigor@sysoev.ru     nxt_python_prepare_msg,
243216Sigor@sysoev.ru     nxt_php_prepare_msg,
244216Sigor@sysoev.ru     nxt_go_prepare_msg,
245216Sigor@sysoev.ru };
246216Sigor@sysoev.ru 
247216Sigor@sysoev.ru 
24820Sigor@sysoev.ru nxt_int_t
249141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
25020Sigor@sysoev.ru {
251141Smax.romanov@nginx.com     nxt_int_t      ret;
252141Smax.romanov@nginx.com     nxt_router_t   *router;
253141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
254141Smax.romanov@nginx.com 
255141Smax.romanov@nginx.com     rt = task->thread->runtime;
25653Sigor@sysoev.ru 
257431Sigor@sysoev.ru     ret = nxt_http_init(task, rt);
25888Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
25988Smax.romanov@nginx.com         return ret;
26088Smax.romanov@nginx.com     }
26188Smax.romanov@nginx.com 
26253Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
26353Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
26453Sigor@sysoev.ru         return NXT_ERROR;
26553Sigor@sysoev.ru     }
26653Sigor@sysoev.ru 
26753Sigor@sysoev.ru     nxt_queue_init(&router->engines);
26853Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
269133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
27053Sigor@sysoev.ru 
271119Smax.romanov@nginx.com     nxt_router = router;
272119Smax.romanov@nginx.com 
273115Sigor@sysoev.ru     return NXT_OK;
274115Sigor@sysoev.ru }
275115Sigor@sysoev.ru 
276115Sigor@sysoev.ru 
277343Smax.romanov@nginx.com static void
278*507Smax.romanov@nginx.com nxt_router_start_app_process_handler(nxt_task_t *task, nxt_port_t *port,
279*507Smax.romanov@nginx.com     void *data)
280167Smax.romanov@nginx.com {
281343Smax.romanov@nginx.com     size_t         size;
282343Smax.romanov@nginx.com     uint32_t       stream;
283430Sigor@sysoev.ru     nxt_mp_t       *mp;
284343Smax.romanov@nginx.com     nxt_app_t      *app;
285343Smax.romanov@nginx.com     nxt_buf_t      *b;
286343Smax.romanov@nginx.com     nxt_port_t     *main_port;
287343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
288343Smax.romanov@nginx.com 
289343Smax.romanov@nginx.com     app = data;
290167Smax.romanov@nginx.com 
291167Smax.romanov@nginx.com     rt = task->thread->runtime;
292240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
293167Smax.romanov@nginx.com 
294*507Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start process", &app->name, app);
295343Smax.romanov@nginx.com 
296343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
297343Smax.romanov@nginx.com 
298343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
299343Smax.romanov@nginx.com 
300343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
301343Smax.romanov@nginx.com         goto failed;
302167Smax.romanov@nginx.com     }
303167Smax.romanov@nginx.com 
304343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
305343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
306343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
307343Smax.romanov@nginx.com 
308343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
309343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
310343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
311343Smax.romanov@nginx.com                                            -1, app);
312343Smax.romanov@nginx.com 
313343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
314430Sigor@sysoev.ru         mp = b->data;
315430Sigor@sysoev.ru         nxt_mp_free(mp, b);
316430Sigor@sysoev.ru         nxt_mp_release(mp);
317343Smax.romanov@nginx.com 
318343Smax.romanov@nginx.com         goto failed;
319343Smax.romanov@nginx.com     }
320343Smax.romanov@nginx.com 
321343Smax.romanov@nginx.com     nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
322343Smax.romanov@nginx.com                           stream, port->id, b);
323343Smax.romanov@nginx.com 
324343Smax.romanov@nginx.com     return;
325343Smax.romanov@nginx.com 
326343Smax.romanov@nginx.com failed:
327343Smax.romanov@nginx.com 
328343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
329343Smax.romanov@nginx.com 
330*507Smax.romanov@nginx.com     app->pending_processes--;
331343Smax.romanov@nginx.com 
332343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
333343Smax.romanov@nginx.com 
334343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
335167Smax.romanov@nginx.com }
336167Smax.romanov@nginx.com 
337167Smax.romanov@nginx.com 
338343Smax.romanov@nginx.com static nxt_int_t
339*507Smax.romanov@nginx.com nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app)
340141Smax.romanov@nginx.com {
341343Smax.romanov@nginx.com     nxt_int_t      res;
342343Smax.romanov@nginx.com     nxt_port_t     *router_port;
343343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
344343Smax.romanov@nginx.com 
345343Smax.romanov@nginx.com     rt = task->thread->runtime;
346343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
347343Smax.romanov@nginx.com 
348343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
349343Smax.romanov@nginx.com 
350*507Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_app_process_handler,
351343Smax.romanov@nginx.com                         app);
352343Smax.romanov@nginx.com 
353343Smax.romanov@nginx.com     if (res == NXT_OK) {
354343Smax.romanov@nginx.com         return res;
355318Smax.romanov@nginx.com     }
356318Smax.romanov@nginx.com 
357343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
358343Smax.romanov@nginx.com 
359*507Smax.romanov@nginx.com     app->pending_processes--;
360343Smax.romanov@nginx.com 
361343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
362343Smax.romanov@nginx.com 
363343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
364343Smax.romanov@nginx.com 
365343Smax.romanov@nginx.com     return NXT_ERROR;
366318Smax.romanov@nginx.com }
367318Smax.romanov@nginx.com 
368318Smax.romanov@nginx.com 
369351Smax.romanov@nginx.com nxt_inline void
370351Smax.romanov@nginx.com nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra,
371351Smax.romanov@nginx.com     nxt_req_conn_link_t *rc)
372167Smax.romanov@nginx.com {
373318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
374351Smax.romanov@nginx.com 
375318Smax.romanov@nginx.com     engine = task->thread->engine;
376167Smax.romanov@nginx.com 
377167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
378167Smax.romanov@nginx.com 
379318Smax.romanov@nginx.com     ra->stream = rc->stream;
380425Smax.romanov@nginx.com     ra->use_count = 1;
381167Smax.romanov@nginx.com     ra->rc = rc;
382318Smax.romanov@nginx.com     rc->ra = ra;
383318Smax.romanov@nginx.com     ra->reply_port = engine->port;
384351Smax.romanov@nginx.com     ra->ap = rc->ap;
385167Smax.romanov@nginx.com 
386167Smax.romanov@nginx.com     ra->work.handler = NULL;
387318Smax.romanov@nginx.com     ra->work.task = &engine->task;
388167Smax.romanov@nginx.com     ra->work.obj = ra;
389318Smax.romanov@nginx.com     ra->work.data = engine;
390351Smax.romanov@nginx.com }
391351Smax.romanov@nginx.com 
392351Smax.romanov@nginx.com 
393351Smax.romanov@nginx.com nxt_inline nxt_req_app_link_t *
394351Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src)
395351Smax.romanov@nginx.com {
396351Smax.romanov@nginx.com     nxt_mp_t            *mp;
397351Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
398351Smax.romanov@nginx.com 
399425Smax.romanov@nginx.com     if (ra_src->mem_pool != NULL) {
400425Smax.romanov@nginx.com         return ra_src;
401425Smax.romanov@nginx.com     }
402425Smax.romanov@nginx.com 
403351Smax.romanov@nginx.com     mp = ra_src->ap->mem_pool;
404351Smax.romanov@nginx.com 
405430Sigor@sysoev.ru     ra = nxt_mp_alloc(mp, sizeof(nxt_req_app_link_t));
406351Smax.romanov@nginx.com 
407351Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
408351Smax.romanov@nginx.com 
409351Smax.romanov@nginx.com         ra_src->rc->ra = NULL;
410351Smax.romanov@nginx.com         ra_src->rc = NULL;
411351Smax.romanov@nginx.com 
412351Smax.romanov@nginx.com         return NULL;
413351Smax.romanov@nginx.com     }
414351Smax.romanov@nginx.com 
415430Sigor@sysoev.ru     nxt_mp_retain(mp);
416430Sigor@sysoev.ru 
417351Smax.romanov@nginx.com     nxt_router_ra_init(task, ra, ra_src->rc);
418351Smax.romanov@nginx.com 
419351Smax.romanov@nginx.com     ra->mem_pool = mp;
420167Smax.romanov@nginx.com 
421167Smax.romanov@nginx.com     return ra;
422167Smax.romanov@nginx.com }
423167Smax.romanov@nginx.com 
424167Smax.romanov@nginx.com 
425423Smax.romanov@nginx.com nxt_inline nxt_bool_t
426423Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info,
427423Smax.romanov@nginx.com     uint32_t stream)
428423Smax.romanov@nginx.com {
429423Smax.romanov@nginx.com     nxt_buf_t   *b, *next;
430423Smax.romanov@nginx.com     nxt_bool_t  cancelled;
431423Smax.romanov@nginx.com 
432423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
433423Smax.romanov@nginx.com         return 0;
434423Smax.romanov@nginx.com     }
435423Smax.romanov@nginx.com 
436423Smax.romanov@nginx.com     cancelled = nxt_port_mmap_tracking_cancel(task, &msg_info->tracking,
437423Smax.romanov@nginx.com                                               stream);
438423Smax.romanov@nginx.com 
439423Smax.romanov@nginx.com     if (cancelled) {
440423Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router", stream);
441423Smax.romanov@nginx.com     }
442423Smax.romanov@nginx.com 
443423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
444423Smax.romanov@nginx.com         next = b->next;
445423Smax.romanov@nginx.com 
446423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
447423Smax.romanov@nginx.com 
448423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
449423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
450423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
451423Smax.romanov@nginx.com         }
452423Smax.romanov@nginx.com     }
453423Smax.romanov@nginx.com 
454423Smax.romanov@nginx.com     msg_info->buf = NULL;
455423Smax.romanov@nginx.com 
456423Smax.romanov@nginx.com     return cancelled;
457423Smax.romanov@nginx.com }
458423Smax.romanov@nginx.com 
459423Smax.romanov@nginx.com 
460167Smax.romanov@nginx.com static void
461425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra);
462425Smax.romanov@nginx.com 
463425Smax.romanov@nginx.com 
464425Smax.romanov@nginx.com static void
465425Smax.romanov@nginx.com nxt_router_ra_update_peer_handler(nxt_task_t *task, void *obj, void *data)
466167Smax.romanov@nginx.com {
467425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
468425Smax.romanov@nginx.com 
469425Smax.romanov@nginx.com     ra = obj;
470425Smax.romanov@nginx.com 
471425Smax.romanov@nginx.com     nxt_router_ra_update_peer(task, ra);
472425Smax.romanov@nginx.com 
473425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
474425Smax.romanov@nginx.com }
475425Smax.romanov@nginx.com 
476425Smax.romanov@nginx.com 
477425Smax.romanov@nginx.com static void
478425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra)
479425Smax.romanov@nginx.com {
480343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
481343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
482318Smax.romanov@nginx.com 
483425Smax.romanov@nginx.com     engine = ra->work.data;
484318Smax.romanov@nginx.com 
485343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
486425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
487425Smax.romanov@nginx.com 
488425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_update_peer_handler;
489318Smax.romanov@nginx.com         ra->work.task = &engine->task;
490318Smax.romanov@nginx.com         ra->work.next = NULL;
491318Smax.romanov@nginx.com 
492425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post update peer to %p",
493318Smax.romanov@nginx.com                   ra->stream, engine);
494318Smax.romanov@nginx.com 
495318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
496318Smax.romanov@nginx.com 
497318Smax.romanov@nginx.com         return;
498318Smax.romanov@nginx.com     }
499318Smax.romanov@nginx.com 
500425Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD update peer", ra->stream);
501425Smax.romanov@nginx.com 
502425Smax.romanov@nginx.com     rc = ra->rc;
503425Smax.romanov@nginx.com 
504425Smax.romanov@nginx.com     if (rc != NULL && ra->app_port != NULL) {
505425Smax.romanov@nginx.com         nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_port->pid);
506425Smax.romanov@nginx.com     }
507425Smax.romanov@nginx.com 
508425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
509425Smax.romanov@nginx.com }
510425Smax.romanov@nginx.com 
511425Smax.romanov@nginx.com 
512425Smax.romanov@nginx.com static void
513425Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, nxt_req_app_link_t *ra)
514425Smax.romanov@nginx.com {
515431Sigor@sysoev.ru     nxt_mp_t                *mp;
516431Sigor@sysoev.ru     nxt_req_conn_link_t     *rc;
517425Smax.romanov@nginx.com 
518425Smax.romanov@nginx.com     nxt_assert(task->thread->engine == ra->work.data);
519425Smax.romanov@nginx.com     nxt_assert(ra->use_count == 0);
520425Smax.romanov@nginx.com 
521343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
522343Smax.romanov@nginx.com 
523343Smax.romanov@nginx.com     rc = ra->rc;
524343Smax.romanov@nginx.com 
525343Smax.romanov@nginx.com     if (rc != NULL) {
526423Smax.romanov@nginx.com         if (nxt_slow_path(ra->err_code != 0)) {
527431Sigor@sysoev.ru             nxt_http_request_error(task, rc->ap->request, ra->err_code);
528423Smax.romanov@nginx.com 
529423Smax.romanov@nginx.com         } else {
530423Smax.romanov@nginx.com             rc->app_port = ra->app_port;
531423Smax.romanov@nginx.com             rc->msg_info = ra->msg_info;
532423Smax.romanov@nginx.com 
533425Smax.romanov@nginx.com             if (rc->app->timeout != 0) {
534431Sigor@sysoev.ru                 rc->ap->timer.handler = nxt_router_app_timeout;
535431Sigor@sysoev.ru                 nxt_timer_add(task->thread->engine, &rc->ap->timer,
536425Smax.romanov@nginx.com                               rc->app->timeout);
537425Smax.romanov@nginx.com             }
538425Smax.romanov@nginx.com 
539423Smax.romanov@nginx.com             ra->app_port = NULL;
540423Smax.romanov@nginx.com             ra->msg_info.buf = NULL;
541423Smax.romanov@nginx.com         }
542343Smax.romanov@nginx.com 
543343Smax.romanov@nginx.com         rc->ra = NULL;
544343Smax.romanov@nginx.com         ra->rc = NULL;
545343Smax.romanov@nginx.com     }
546343Smax.romanov@nginx.com 
547343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
548343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
549343Smax.romanov@nginx.com 
550343Smax.romanov@nginx.com         ra->app_port = NULL;
551167Smax.romanov@nginx.com     }
552167Smax.romanov@nginx.com 
553423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &ra->msg_info, ra->stream);
554423Smax.romanov@nginx.com 
555430Sigor@sysoev.ru     mp = ra->mem_pool;
556430Sigor@sysoev.ru 
557430Sigor@sysoev.ru     if (mp != NULL) {
558430Sigor@sysoev.ru         nxt_mp_free(mp, ra);
559430Sigor@sysoev.ru         nxt_mp_release(mp);
560351Smax.romanov@nginx.com     }
561167Smax.romanov@nginx.com }
562167Smax.romanov@nginx.com 
563167Smax.romanov@nginx.com 
564425Smax.romanov@nginx.com static void
565425Smax.romanov@nginx.com nxt_router_ra_release_handler(nxt_task_t *task, void *obj, void *data)
566425Smax.romanov@nginx.com {
567425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
568425Smax.romanov@nginx.com 
569425Smax.romanov@nginx.com     ra = obj;
570425Smax.romanov@nginx.com 
571425Smax.romanov@nginx.com     nxt_assert(ra->work.data == data);
572425Smax.romanov@nginx.com 
573425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, -1);
574425Smax.romanov@nginx.com 
575425Smax.romanov@nginx.com     nxt_router_ra_release(task, ra);
576425Smax.romanov@nginx.com }
577425Smax.romanov@nginx.com 
578425Smax.romanov@nginx.com 
579425Smax.romanov@nginx.com static void
580425Smax.romanov@nginx.com nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i)
581425Smax.romanov@nginx.com {
582425Smax.romanov@nginx.com     int                 c;
583425Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
584425Smax.romanov@nginx.com 
585425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, i);
586425Smax.romanov@nginx.com 
587425Smax.romanov@nginx.com     if (i < 0 && c == -i) {
588425Smax.romanov@nginx.com         engine = ra->work.data;
589425Smax.romanov@nginx.com 
590425Smax.romanov@nginx.com         if (task->thread->engine == engine) {
591425Smax.romanov@nginx.com             nxt_router_ra_release(task, ra);
592425Smax.romanov@nginx.com 
593425Smax.romanov@nginx.com             return;
594425Smax.romanov@nginx.com         }
595425Smax.romanov@nginx.com 
596425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
597425Smax.romanov@nginx.com 
598425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release_handler;
599425Smax.romanov@nginx.com         ra->work.task = &engine->task;
600425Smax.romanov@nginx.com         ra->work.next = NULL;
601425Smax.romanov@nginx.com 
602425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
603425Smax.romanov@nginx.com                   ra->stream, engine);
604425Smax.romanov@nginx.com 
605425Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
606425Smax.romanov@nginx.com     }
607425Smax.romanov@nginx.com }
608425Smax.romanov@nginx.com 
609425Smax.romanov@nginx.com 
610423Smax.romanov@nginx.com nxt_inline void
611423Smax.romanov@nginx.com nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str)
612345Smax.romanov@nginx.com {
613423Smax.romanov@nginx.com     ra->app_port = NULL;
614423Smax.romanov@nginx.com     ra->err_code = code;
615423Smax.romanov@nginx.com     ra->err_str = str;
616345Smax.romanov@nginx.com }
617345Smax.romanov@nginx.com 
618345Smax.romanov@nginx.com 
619427Smax.romanov@nginx.com nxt_inline void
620427Smax.romanov@nginx.com nxt_router_ra_pending(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
621427Smax.romanov@nginx.com {
622427Smax.romanov@nginx.com     nxt_queue_insert_tail(&ra->app_port->pending_requests,
623427Smax.romanov@nginx.com                           &ra->link_port_pending);
624427Smax.romanov@nginx.com     nxt_queue_insert_tail(&app->pending, &ra->link_app_pending);
625427Smax.romanov@nginx.com 
626427Smax.romanov@nginx.com     nxt_router_ra_inc_use(ra);
627427Smax.romanov@nginx.com 
628427Smax.romanov@nginx.com     ra->res_time = nxt_thread_monotonic_time(task->thread) + app->res_timeout;
629427Smax.romanov@nginx.com 
630427Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD enqueue to pending_requests", ra->stream);
631427Smax.romanov@nginx.com }
632427Smax.romanov@nginx.com 
633427Smax.romanov@nginx.com 
634425Smax.romanov@nginx.com nxt_inline nxt_bool_t
635425Smax.romanov@nginx.com nxt_queue_chk_remove(nxt_queue_link_t *lnk)
636425Smax.romanov@nginx.com {
637425Smax.romanov@nginx.com     if (lnk->next != NULL) {
638425Smax.romanov@nginx.com         nxt_queue_remove(lnk);
639425Smax.romanov@nginx.com 
640425Smax.romanov@nginx.com         lnk->next = NULL;
641425Smax.romanov@nginx.com 
642425Smax.romanov@nginx.com         return 1;
643425Smax.romanov@nginx.com     }
644425Smax.romanov@nginx.com 
645425Smax.romanov@nginx.com     return 0;
646425Smax.romanov@nginx.com }
647425Smax.romanov@nginx.com 
648425Smax.romanov@nginx.com 
649343Smax.romanov@nginx.com nxt_inline void
650343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
651343Smax.romanov@nginx.com {
652425Smax.romanov@nginx.com     int                 ra_use_delta;
653343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
654343Smax.romanov@nginx.com 
655343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
656343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
657343Smax.romanov@nginx.com 
658343Smax.romanov@nginx.com         rc->app_port = NULL;
659343Smax.romanov@nginx.com     }
660343Smax.romanov@nginx.com 
661423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &rc->msg_info, rc->stream);
662423Smax.romanov@nginx.com 
663343Smax.romanov@nginx.com     ra = rc->ra;
664343Smax.romanov@nginx.com 
665343Smax.romanov@nginx.com     if (ra != NULL) {
666343Smax.romanov@nginx.com         rc->ra = NULL;
667343Smax.romanov@nginx.com         ra->rc = NULL;
668343Smax.romanov@nginx.com 
669425Smax.romanov@nginx.com         ra_use_delta = 0;
670425Smax.romanov@nginx.com 
671343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
672343Smax.romanov@nginx.com 
673425Smax.romanov@nginx.com         if (ra->link_app_requests.next == NULL
674427Smax.romanov@nginx.com             && ra->link_port_pending.next == NULL
675427Smax.romanov@nginx.com             && ra->link_app_pending.next == NULL)
676425Smax.romanov@nginx.com         {
677425Smax.romanov@nginx.com             ra = NULL;
678343Smax.romanov@nginx.com 
679343Smax.romanov@nginx.com         } else {
680425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests);
681425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending);
682427Smax.romanov@nginx.com             nxt_queue_chk_remove(&ra->link_app_pending);
683343Smax.romanov@nginx.com         }
684343Smax.romanov@nginx.com 
685343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
686425Smax.romanov@nginx.com 
687425Smax.romanov@nginx.com         if (ra != NULL) {
688425Smax.romanov@nginx.com             nxt_router_ra_use(task, ra, ra_use_delta);
689425Smax.romanov@nginx.com         }
690343Smax.romanov@nginx.com     }
691343Smax.romanov@nginx.com 
692343Smax.romanov@nginx.com     if (rc->app != NULL) {
693343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
694343Smax.romanov@nginx.com 
695343Smax.romanov@nginx.com         rc->app = NULL;
696343Smax.romanov@nginx.com     }
697343Smax.romanov@nginx.com 
698346Smax.romanov@nginx.com     if (rc->ap != NULL) {
699346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
700346Smax.romanov@nginx.com 
701346Smax.romanov@nginx.com         rc->ap = NULL;
702346Smax.romanov@nginx.com     }
703343Smax.romanov@nginx.com }
704343Smax.romanov@nginx.com 
705343Smax.romanov@nginx.com 
706141Smax.romanov@nginx.com void
707141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
708141Smax.romanov@nginx.com {
709141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
710141Smax.romanov@nginx.com 
711192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
712141Smax.romanov@nginx.com         return;
713141Smax.romanov@nginx.com     }
714141Smax.romanov@nginx.com 
715426Smax.romanov@nginx.com     if (msg->u.new_port == NULL
716426Smax.romanov@nginx.com         || msg->u.new_port->type != NXT_PROCESS_WORKER)
717347Smax.romanov@nginx.com     {
718192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
719141Smax.romanov@nginx.com     }
720192Smax.romanov@nginx.com 
721192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
722141Smax.romanov@nginx.com }
723141Smax.romanov@nginx.com 
724141Smax.romanov@nginx.com 
725139Sigor@sysoev.ru void
726139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
727115Sigor@sysoev.ru {
728198Sigor@sysoev.ru     nxt_int_t               ret;
729139Sigor@sysoev.ru     nxt_buf_t               *b;
730139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
731139Sigor@sysoev.ru 
732139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
733139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
734139Sigor@sysoev.ru         return;
73553Sigor@sysoev.ru     }
73653Sigor@sysoev.ru 
737494Spluknet@nginx.com     nxt_debug(task, "nxt_router_conf_data_handler(%O): %*s",
738423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf),
739493Spluknet@nginx.com               (size_t) nxt_buf_used_size(msg->buf), msg->buf->mem.pos);
740423Smax.romanov@nginx.com 
741352Smax.romanov@nginx.com     b = nxt_buf_chk_make_plain(tmcf->conf->mem_pool, msg->buf, msg->size);
742352Smax.romanov@nginx.com 
743352Smax.romanov@nginx.com     nxt_assert(b != NULL);
744352Smax.romanov@nginx.com 
745139Sigor@sysoev.ru     tmcf->conf->router = nxt_router;
746139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
747139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
748198Sigor@sysoev.ru                                        msg->port_msg.pid,
749198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
750198Sigor@sysoev.ru 
751198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
752198Sigor@sysoev.ru 
753198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
754198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
755198Sigor@sysoev.ru 
756198Sigor@sysoev.ru     } else {
757198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
758139Sigor@sysoev.ru     }
75953Sigor@sysoev.ru }
76053Sigor@sysoev.ru 
76153Sigor@sysoev.ru 
762347Smax.romanov@nginx.com static void
763*507Smax.romanov@nginx.com nxt_router_app_process_remove_pid(nxt_task_t *task, nxt_port_t *port,
764*507Smax.romanov@nginx.com     void *data)
765347Smax.romanov@nginx.com {
766347Smax.romanov@nginx.com     union {
767347Smax.romanov@nginx.com         nxt_pid_t  removed_pid;
768347Smax.romanov@nginx.com         void       *data;
769347Smax.romanov@nginx.com     } u;
770347Smax.romanov@nginx.com 
771347Smax.romanov@nginx.com     u.data = data;
772347Smax.romanov@nginx.com 
773347Smax.romanov@nginx.com     nxt_port_rpc_remove_peer(task, port, u.removed_pid);
774347Smax.romanov@nginx.com }
775347Smax.romanov@nginx.com 
776347Smax.romanov@nginx.com 
777192Smax.romanov@nginx.com void
778192Smax.romanov@nginx.com nxt_router_remove_pid_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
779192Smax.romanov@nginx.com {
780347Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
781318Smax.romanov@nginx.com 
782192Smax.romanov@nginx.com     nxt_port_remove_pid_handler(task, msg);
783192Smax.romanov@nginx.com 
784192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
785192Smax.romanov@nginx.com         return;
786192Smax.romanov@nginx.com     }
787192Smax.romanov@nginx.com 
788318Smax.romanov@nginx.com     nxt_queue_each(engine, &nxt_router->engines, nxt_event_engine_t, link0)
789318Smax.romanov@nginx.com     {
790*507Smax.romanov@nginx.com         nxt_port_post(task, engine->port, nxt_router_app_process_remove_pid,
791347Smax.romanov@nginx.com                       msg->u.data);
792318Smax.romanov@nginx.com     }
793318Smax.romanov@nginx.com