xref: /unit/src/nxt_router.c (revision 423)
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>
1020Sigor@sysoev.ru 
1120Sigor@sysoev.ru 
12115Sigor@sysoev.ru typedef struct {
13318Smax.romanov@nginx.com     nxt_str_t         type;
14318Smax.romanov@nginx.com     uint32_t          workers;
15318Smax.romanov@nginx.com     nxt_msec_t        timeout;
16318Smax.romanov@nginx.com     uint32_t          requests;
17318Smax.romanov@nginx.com     nxt_conf_value_t  *limits_value;
18133Sigor@sysoev.ru } nxt_router_app_conf_t;
19133Sigor@sysoev.ru 
20133Sigor@sysoev.ru 
21133Sigor@sysoev.ru typedef struct {
22133Sigor@sysoev.ru     nxt_str_t  application;
23115Sigor@sysoev.ru } nxt_router_listener_conf_t;
24115Sigor@sysoev.ru 
25115Sigor@sysoev.ru 
26*423Smax.romanov@nginx.com typedef struct nxt_msg_info_s {
27*423Smax.romanov@nginx.com     nxt_buf_t                 *buf;
28*423Smax.romanov@nginx.com     nxt_port_mmap_tracking_t  tracking;
29*423Smax.romanov@nginx.com     nxt_work_handler_t        completion_handler;
30*423Smax.romanov@nginx.com } nxt_msg_info_t;
31*423Smax.romanov@nginx.com 
32*423Smax.romanov@nginx.com 
33167Smax.romanov@nginx.com typedef struct nxt_req_app_link_s nxt_req_app_link_t;
34141Smax.romanov@nginx.com 
35141Smax.romanov@nginx.com 
36318Smax.romanov@nginx.com typedef struct {
37318Smax.romanov@nginx.com     uint32_t             stream;
38318Smax.romanov@nginx.com     nxt_conn_t           *conn;
39343Smax.romanov@nginx.com     nxt_app_t            *app;
40318Smax.romanov@nginx.com     nxt_port_t           *app_port;
41346Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
42*423Smax.romanov@nginx.com     nxt_msg_info_t       msg_info;
43318Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
44318Smax.romanov@nginx.com 
45318Smax.romanov@nginx.com     nxt_queue_link_t     link;     /* for nxt_conn_t.requests */
46318Smax.romanov@nginx.com } nxt_req_conn_link_t;
47318Smax.romanov@nginx.com 
48318Smax.romanov@nginx.com 
49167Smax.romanov@nginx.com struct nxt_req_app_link_s {
50318Smax.romanov@nginx.com     uint32_t             stream;
51167Smax.romanov@nginx.com     nxt_port_t           *app_port;
52318Smax.romanov@nginx.com     nxt_pid_t            app_pid;
53167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
54167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
55*423Smax.romanov@nginx.com     nxt_msg_info_t       msg_info;
56167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
57167Smax.romanov@nginx.com 
58167Smax.romanov@nginx.com     nxt_queue_link_t     link; /* for nxt_app_t.requests */
59167Smax.romanov@nginx.com 
60167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
61167Smax.romanov@nginx.com     nxt_work_t           work;
62345Smax.romanov@nginx.com 
63345Smax.romanov@nginx.com     int                  err_code;
64345Smax.romanov@nginx.com     const char           *err_str;
65167Smax.romanov@nginx.com };
66167Smax.romanov@nginx.com 
67167Smax.romanov@nginx.com 
68198Sigor@sysoev.ru typedef struct {
69198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
70198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
71198Sigor@sysoev.ru } nxt_socket_rpc_t;
72198Sigor@sysoev.ru 
73198Sigor@sysoev.ru 
74343Smax.romanov@nginx.com static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app);
75343Smax.romanov@nginx.com 
76139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
77198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
78198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
79139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
80139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
81139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
82139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
83193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
8453Sigor@sysoev.ru 
85115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
86115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
87133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
88133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
89133Sigor@sysoev.ru     nxt_str_t *name);
90198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
91198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
92198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
93198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
94198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
95198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
96359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
97359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
98359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
99359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
10053Sigor@sysoev.ru 
10153Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
10253Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
10353Sigor@sysoev.ru     const nxt_event_interface_t *interface);
104115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
105115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
106115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
107115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
108115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
109115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
110154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
111154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
112154Sigor@sysoev.ru     nxt_work_handler_t handler);
113313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
114313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
115139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
116139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
11753Sigor@sysoev.ru 
11853Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
11953Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
12053Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
12153Sigor@sysoev.ru     nxt_event_engine_t *engine);
122343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
123133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
12453Sigor@sysoev.ru 
125315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
126315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
127315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
128315Sigor@sysoev.ru     nxt_work_t *jobs);
12953Sigor@sysoev.ru 
13053Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
13153Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
13253Sigor@sysoev.ru     void *data);
13353Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
13453Sigor@sysoev.ru     void *data);
13553Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
13653Sigor@sysoev.ru     void *data);
137313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
138313Sigor@sysoev.ru     void *data);
13953Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
14053Sigor@sysoev.ru     void *data);
14153Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
14253Sigor@sysoev.ru     void *data);
143359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
144359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
14553Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
14653Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
14753Sigor@sysoev.ru 
148343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
149343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
150343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
151343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
152343Smax.romanov@nginx.com 
153343Smax.romanov@nginx.com static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app);
154343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
155343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
156141Smax.romanov@nginx.com 
15753Sigor@sysoev.ru static void nxt_router_conn_init(nxt_task_t *task, void *obj, void *data);
15853Sigor@sysoev.ru static void nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj,
15953Sigor@sysoev.ru     void *data);
160359Sigor@sysoev.ru static nxt_sockaddr_t *nxt_router_local_addr(nxt_task_t *task, nxt_conn_t *c);
161206Smax.romanov@nginx.com static void nxt_router_conn_http_body_read(nxt_task_t *task, void *obj,
162206Smax.romanov@nginx.com     void *data);
16388Smax.romanov@nginx.com static void nxt_router_process_http_request(nxt_task_t *task,
16488Smax.romanov@nginx.com     nxt_conn_t *c, nxt_app_parse_ctx_t *ap);
165141Smax.romanov@nginx.com static void nxt_router_process_http_request_mp(nxt_task_t *task,
166343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
167216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
168216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
169216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
170216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
171216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
172216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
17388Smax.romanov@nginx.com static void nxt_router_conn_ready(nxt_task_t *task, void *obj, void *data);
17453Sigor@sysoev.ru static void nxt_router_conn_close(nxt_task_t *task, void *obj, void *data);
17553Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
17653Sigor@sysoev.ru static void nxt_router_conn_error(nxt_task_t *task, void *obj, void *data);
17753Sigor@sysoev.ru static void nxt_router_conn_timeout(nxt_task_t *task, void *obj, void *data);
178318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
17962Sigor@sysoev.ru static nxt_msec_t nxt_router_conn_timeout_value(nxt_conn_t *c, uintptr_t data);
18020Sigor@sysoev.ru 
181141Smax.romanov@nginx.com static void nxt_router_gen_error(nxt_task_t *task, nxt_conn_t *c, int code,
182345Smax.romanov@nginx.com     const char* str);
183141Smax.romanov@nginx.com 
184119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
18520Sigor@sysoev.ru 
186216Sigor@sysoev.ru 
187216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
188216Sigor@sysoev.ru     nxt_python_prepare_msg,
189216Sigor@sysoev.ru     nxt_php_prepare_msg,
190216Sigor@sysoev.ru     nxt_go_prepare_msg,
191216Sigor@sysoev.ru };
192216Sigor@sysoev.ru 
193216Sigor@sysoev.ru 
19420Sigor@sysoev.ru nxt_int_t
195141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
19620Sigor@sysoev.ru {
197141Smax.romanov@nginx.com     nxt_int_t      ret;
198141Smax.romanov@nginx.com     nxt_router_t   *router;
199141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
200141Smax.romanov@nginx.com 
201141Smax.romanov@nginx.com     rt = task->thread->runtime;
20253Sigor@sysoev.ru 
20388Smax.romanov@nginx.com     ret = nxt_app_http_init(task, rt);
20488Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
20588Smax.romanov@nginx.com         return ret;
20688Smax.romanov@nginx.com     }
20788Smax.romanov@nginx.com 
20853Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
20953Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
21053Sigor@sysoev.ru         return NXT_ERROR;
21153Sigor@sysoev.ru     }
21253Sigor@sysoev.ru 
21353Sigor@sysoev.ru     nxt_queue_init(&router->engines);
21453Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
215133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
21653Sigor@sysoev.ru 
217119Smax.romanov@nginx.com     nxt_router = router;
218119Smax.romanov@nginx.com 
219115Sigor@sysoev.ru     return NXT_OK;
220115Sigor@sysoev.ru }
221115Sigor@sysoev.ru 
222115Sigor@sysoev.ru 
223343Smax.romanov@nginx.com static void
224343Smax.romanov@nginx.com nxt_router_start_worker_handler(nxt_task_t *task, nxt_port_t *port, void *data)
225167Smax.romanov@nginx.com {
226343Smax.romanov@nginx.com     size_t         size;
227343Smax.romanov@nginx.com     uint32_t       stream;
228343Smax.romanov@nginx.com     nxt_app_t      *app;
229343Smax.romanov@nginx.com     nxt_buf_t      *b;
230343Smax.romanov@nginx.com     nxt_port_t     *main_port;
231343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
232343Smax.romanov@nginx.com 
233343Smax.romanov@nginx.com     app = data;
234167Smax.romanov@nginx.com 
235167Smax.romanov@nginx.com     rt = task->thread->runtime;
236240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
237167Smax.romanov@nginx.com 
238343Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start worker", &app->name, app);
239343Smax.romanov@nginx.com 
240343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
241343Smax.romanov@nginx.com 
242343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
243343Smax.romanov@nginx.com 
244343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
245343Smax.romanov@nginx.com         goto failed;
246167Smax.romanov@nginx.com     }
247167Smax.romanov@nginx.com 
248343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
249343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
250343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
251343Smax.romanov@nginx.com 
252343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
253343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
254343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
255343Smax.romanov@nginx.com                                            -1, app);
256343Smax.romanov@nginx.com 
257343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
258343Smax.romanov@nginx.com         nxt_mp_release(b->data, b);
259343Smax.romanov@nginx.com 
260343Smax.romanov@nginx.com         goto failed;
261343Smax.romanov@nginx.com     }
262343Smax.romanov@nginx.com 
263343Smax.romanov@nginx.com     nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
264343Smax.romanov@nginx.com                           stream, port->id, b);
265343Smax.romanov@nginx.com 
266343Smax.romanov@nginx.com     return;
267343Smax.romanov@nginx.com 
268343Smax.romanov@nginx.com failed:
269343Smax.romanov@nginx.com 
270343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
271343Smax.romanov@nginx.com 
272343Smax.romanov@nginx.com     app->pending_workers--;
273343Smax.romanov@nginx.com 
274343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
275343Smax.romanov@nginx.com 
276343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
277167Smax.romanov@nginx.com }
278167Smax.romanov@nginx.com 
279167Smax.romanov@nginx.com 
280343Smax.romanov@nginx.com static nxt_int_t
281343Smax.romanov@nginx.com nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app)
282141Smax.romanov@nginx.com {
283343Smax.romanov@nginx.com     nxt_int_t      res;
284343Smax.romanov@nginx.com     nxt_port_t     *router_port;
285343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
286343Smax.romanov@nginx.com 
287343Smax.romanov@nginx.com     rt = task->thread->runtime;
288343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
289343Smax.romanov@nginx.com 
290343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
291343Smax.romanov@nginx.com 
292343Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_worker_handler,
293343Smax.romanov@nginx.com                         app);
294343Smax.romanov@nginx.com 
295343Smax.romanov@nginx.com     if (res == NXT_OK) {
296343Smax.romanov@nginx.com         return res;
297318Smax.romanov@nginx.com     }
298318Smax.romanov@nginx.com 
299343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
300343Smax.romanov@nginx.com 
301343Smax.romanov@nginx.com     app->pending_workers--;
302343Smax.romanov@nginx.com 
303343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
304343Smax.romanov@nginx.com 
305343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
306343Smax.romanov@nginx.com 
307343Smax.romanov@nginx.com     return NXT_ERROR;
308318Smax.romanov@nginx.com }
309318Smax.romanov@nginx.com 
310318Smax.romanov@nginx.com 
311351Smax.romanov@nginx.com nxt_inline void
312351Smax.romanov@nginx.com nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra,
313351Smax.romanov@nginx.com     nxt_req_conn_link_t *rc)
314167Smax.romanov@nginx.com {
315318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
316351Smax.romanov@nginx.com 
317318Smax.romanov@nginx.com     engine = task->thread->engine;
318167Smax.romanov@nginx.com 
319167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
320167Smax.romanov@nginx.com 
321318Smax.romanov@nginx.com     ra->stream = rc->stream;
322318Smax.romanov@nginx.com     ra->app_pid = -1;
323167Smax.romanov@nginx.com     ra->rc = rc;
324318Smax.romanov@nginx.com     rc->ra = ra;
325318Smax.romanov@nginx.com     ra->reply_port = engine->port;
326351Smax.romanov@nginx.com     ra->ap = rc->ap;
327167Smax.romanov@nginx.com 
328167Smax.romanov@nginx.com     ra->work.handler = NULL;
329318Smax.romanov@nginx.com     ra->work.task = &engine->task;
330167Smax.romanov@nginx.com     ra->work.obj = ra;
331318Smax.romanov@nginx.com     ra->work.data = engine;
332351Smax.romanov@nginx.com }
333351Smax.romanov@nginx.com 
334351Smax.romanov@nginx.com 
335351Smax.romanov@nginx.com nxt_inline nxt_req_app_link_t *
336351Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src)
337351Smax.romanov@nginx.com {
338351Smax.romanov@nginx.com     nxt_mp_t            *mp;
339351Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
340351Smax.romanov@nginx.com 
341351Smax.romanov@nginx.com     mp = ra_src->ap->mem_pool;
342351Smax.romanov@nginx.com 
343351Smax.romanov@nginx.com     ra = nxt_mp_retain(mp, sizeof(nxt_req_app_link_t));
344351Smax.romanov@nginx.com 
345351Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
346351Smax.romanov@nginx.com 
347351Smax.romanov@nginx.com         ra_src->rc->ra = NULL;
348351Smax.romanov@nginx.com         ra_src->rc = NULL;
349351Smax.romanov@nginx.com 
350351Smax.romanov@nginx.com         return NULL;
351351Smax.romanov@nginx.com     }
352351Smax.romanov@nginx.com 
353351Smax.romanov@nginx.com     nxt_router_ra_init(task, ra, ra_src->rc);
354351Smax.romanov@nginx.com 
355351Smax.romanov@nginx.com     ra->mem_pool = mp;
356167Smax.romanov@nginx.com 
357167Smax.romanov@nginx.com     return ra;
358167Smax.romanov@nginx.com }
359167Smax.romanov@nginx.com 
360167Smax.romanov@nginx.com 
361*423Smax.romanov@nginx.com nxt_inline nxt_bool_t
362*423Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info,
363*423Smax.romanov@nginx.com     uint32_t stream)
364*423Smax.romanov@nginx.com {
365*423Smax.romanov@nginx.com     nxt_buf_t   *b, *next;
366*423Smax.romanov@nginx.com     nxt_bool_t  cancelled;
367*423Smax.romanov@nginx.com 
368*423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
369*423Smax.romanov@nginx.com         return 0;
370*423Smax.romanov@nginx.com     }
371*423Smax.romanov@nginx.com 
372*423Smax.romanov@nginx.com     cancelled = nxt_port_mmap_tracking_cancel(task, &msg_info->tracking,
373*423Smax.romanov@nginx.com                                               stream);
374*423Smax.romanov@nginx.com 
375*423Smax.romanov@nginx.com     if (cancelled) {
376*423Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router", stream);
377*423Smax.romanov@nginx.com     }
378*423Smax.romanov@nginx.com 
379*423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
380*423Smax.romanov@nginx.com         next = b->next;
381*423Smax.romanov@nginx.com 
382*423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
383*423Smax.romanov@nginx.com 
384*423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
385*423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
386*423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
387*423Smax.romanov@nginx.com         }
388*423Smax.romanov@nginx.com     }
389*423Smax.romanov@nginx.com 
390*423Smax.romanov@nginx.com     msg_info->buf = NULL;
391*423Smax.romanov@nginx.com 
392*423Smax.romanov@nginx.com     return cancelled;
393*423Smax.romanov@nginx.com }
394*423Smax.romanov@nginx.com 
395*423Smax.romanov@nginx.com 
396167Smax.romanov@nginx.com static void
397167Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, void *obj, void *data)
398167Smax.romanov@nginx.com {
399343Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
400343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
401343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
402318Smax.romanov@nginx.com 
403318Smax.romanov@nginx.com     ra = obj;
404318Smax.romanov@nginx.com     engine = data;
405318Smax.romanov@nginx.com 
406343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
407343Smax.romanov@nginx.com         if (ra->app_port != NULL) {
408343Smax.romanov@nginx.com             ra->app_pid = ra->app_port->pid;
409318Smax.romanov@nginx.com         }
410318Smax.romanov@nginx.com 
411318Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release;
412318Smax.romanov@nginx.com         ra->work.task = &engine->task;
413318Smax.romanov@nginx.com         ra->work.next = NULL;
414318Smax.romanov@nginx.com 
415318Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
416318Smax.romanov@nginx.com                   ra->stream, engine);
417318Smax.romanov@nginx.com 
418318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
419318Smax.romanov@nginx.com 
420318Smax.romanov@nginx.com         return;
421318Smax.romanov@nginx.com     }
422318Smax.romanov@nginx.com 
423343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
424343Smax.romanov@nginx.com 
425343Smax.romanov@nginx.com     rc = ra->rc;
426343Smax.romanov@nginx.com 
427343Smax.romanov@nginx.com     if (rc != NULL) {
428343Smax.romanov@nginx.com         if (ra->app_pid != -1) {
429343Smax.romanov@nginx.com             nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_pid);
430343Smax.romanov@nginx.com         }
431343Smax.romanov@nginx.com 
432*423Smax.romanov@nginx.com         if (nxt_slow_path(ra->err_code != 0)) {
433*423Smax.romanov@nginx.com             nxt_router_gen_error(task, rc->conn, ra->err_code, ra->err_str);
434*423Smax.romanov@nginx.com 
435*423Smax.romanov@nginx.com         } else {
436*423Smax.romanov@nginx.com             rc->app_port = ra->app_port;
437*423Smax.romanov@nginx.com             rc->msg_info = ra->msg_info;
438*423Smax.romanov@nginx.com 
439*423Smax.romanov@nginx.com             ra->app_port = NULL;
440*423Smax.romanov@nginx.com             ra->msg_info.buf = NULL;
441*423Smax.romanov@nginx.com         }
442343Smax.romanov@nginx.com 
443343Smax.romanov@nginx.com         rc->ra = NULL;
444343Smax.romanov@nginx.com         ra->rc = NULL;
445343Smax.romanov@nginx.com     }
446343Smax.romanov@nginx.com 
447343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
448343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
449343Smax.romanov@nginx.com 
450343Smax.romanov@nginx.com         ra->app_port = NULL;
451167Smax.romanov@nginx.com     }
452167Smax.romanov@nginx.com 
453*423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &ra->msg_info, ra->stream);
454*423Smax.romanov@nginx.com 
455351Smax.romanov@nginx.com     if (ra->mem_pool != NULL) {
456351Smax.romanov@nginx.com         nxt_mp_release(ra->mem_pool, ra);
457351Smax.romanov@nginx.com     }
458167Smax.romanov@nginx.com }
459167Smax.romanov@nginx.com 
460167Smax.romanov@nginx.com 
461*423Smax.romanov@nginx.com nxt_inline void
462*423Smax.romanov@nginx.com nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char* str)
463345Smax.romanov@nginx.com {
464*423Smax.romanov@nginx.com     ra->app_port = NULL;
465*423Smax.romanov@nginx.com     ra->err_code = code;
466*423Smax.romanov@nginx.com     ra->err_str = str;
467345Smax.romanov@nginx.com }
468345Smax.romanov@nginx.com 
469345Smax.romanov@nginx.com 
470343Smax.romanov@nginx.com nxt_inline void
471343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
472343Smax.romanov@nginx.com {
473343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
474343Smax.romanov@nginx.com 
475343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
476343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
477343Smax.romanov@nginx.com 
478343Smax.romanov@nginx.com         rc->app_port = NULL;
479343Smax.romanov@nginx.com     }
480343Smax.romanov@nginx.com 
481*423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &rc->msg_info, rc->stream);
482*423Smax.romanov@nginx.com 
483343Smax.romanov@nginx.com     ra = rc->ra;
484343Smax.romanov@nginx.com 
485343Smax.romanov@nginx.com     if (ra != NULL) {
486343Smax.romanov@nginx.com         rc->ra = NULL;
487343Smax.romanov@nginx.com         ra->rc = NULL;
488343Smax.romanov@nginx.com 
489343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
490343Smax.romanov@nginx.com 
491343Smax.romanov@nginx.com         if (ra->link.next != NULL) {
492343Smax.romanov@nginx.com             nxt_queue_remove(&ra->link);
493343Smax.romanov@nginx.com 
494343Smax.romanov@nginx.com             ra->link.next = NULL;
495343Smax.romanov@nginx.com 
496343Smax.romanov@nginx.com         } else {
497343Smax.romanov@nginx.com             ra = NULL;
498343Smax.romanov@nginx.com         }
499343Smax.romanov@nginx.com 
500343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
501343Smax.romanov@nginx.com     }
502343Smax.romanov@nginx.com 
503343Smax.romanov@nginx.com     if (ra != NULL) {
504343Smax.romanov@nginx.com         nxt_router_ra_release(task, ra, ra->work.data);
505343Smax.romanov@nginx.com     }
506343Smax.romanov@nginx.com 
507343Smax.romanov@nginx.com     if (rc->app != NULL) {
508343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
509343Smax.romanov@nginx.com 
510343Smax.romanov@nginx.com         rc->app = NULL;
511343Smax.romanov@nginx.com     }
512343Smax.romanov@nginx.com 
513346Smax.romanov@nginx.com     if (rc->ap != NULL) {
514346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
515346Smax.romanov@nginx.com 
516346Smax.romanov@nginx.com         rc->ap = NULL;
517346Smax.romanov@nginx.com     }
518346Smax.romanov@nginx.com 
519343Smax.romanov@nginx.com     nxt_queue_remove(&rc->link);
520343Smax.romanov@nginx.com 
521343Smax.romanov@nginx.com     rc->conn = NULL;
522343Smax.romanov@nginx.com }
523343Smax.romanov@nginx.com 
524343Smax.romanov@nginx.com 
525141Smax.romanov@nginx.com void
526141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
527141Smax.romanov@nginx.com {
528141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
529141Smax.romanov@nginx.com 
530192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
531141Smax.romanov@nginx.com         return;
532141Smax.romanov@nginx.com     }
533141Smax.romanov@nginx.com 
534347Smax.romanov@nginx.com     if (msg->u.new_port == NULL ||
535347Smax.romanov@nginx.com         msg->u.new_port->type != NXT_PROCESS_WORKER)
536347Smax.romanov@nginx.com     {
537192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
538141Smax.romanov@nginx.com     }
539192Smax.romanov@nginx.com 
540192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
541141Smax.romanov@nginx.com }
542141Smax.romanov@nginx.com 
543141Smax.romanov@nginx.com 
544139Sigor@sysoev.ru void
545139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
546115Sigor@sysoev.ru {
547198Sigor@sysoev.ru     nxt_int_t               ret;
548139Sigor@sysoev.ru     nxt_buf_t               *b;
549139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
550139Sigor@sysoev.ru 
551139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
552139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
553139Sigor@sysoev.ru         return;
55453Sigor@sysoev.ru     }
55553Sigor@sysoev.ru 
556*423Smax.romanov@nginx.com     nxt_debug(task, "nxt_router_conf_data_handler(%d): %*s",
557*423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf),
558*423Smax.romanov@nginx.com               nxt_buf_used_size(msg->buf), msg->buf->mem.pos);
559*423Smax.romanov@nginx.com 
560352Smax.romanov@nginx.com     b = nxt_buf_chk_make_plain(tmcf->conf->mem_pool, msg->buf, msg->size);
561352Smax.romanov@nginx.com 
562352Smax.romanov@nginx.com     nxt_assert(b != NULL);
563352Smax.romanov@nginx.com 
564139Sigor@sysoev.ru     tmcf->conf->router = nxt_router;
565139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
566139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
567198Sigor@sysoev.ru                                        msg->port_msg.pid,
568198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
569198Sigor@sysoev.ru 
570198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
571198Sigor@sysoev.ru 
572198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
573198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
574198Sigor@sysoev.ru 
575198Sigor@sysoev.ru     } else {
576198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
577139Sigor@sysoev.ru     }
57853Sigor@sysoev.ru }
57953Sigor@sysoev.ru 
58053Sigor@sysoev.ru 
581347Smax.romanov@nginx.com static void
582347Smax.romanov@nginx.com nxt_router_worker_remove_pid(nxt_task_t *task, nxt_port_t *port, void *data)
583347Smax.romanov@nginx.com {
584347Smax.romanov@nginx.com     union {
585347Smax.romanov@nginx.com         nxt_pid_t  removed_pid;
586347Smax.romanov@nginx.com         void       *data;
587347Smax.romanov@nginx.com     } u;
588347Smax.romanov@nginx.com 
589347Smax.romanov@nginx.com     u.data = data;
590347Smax.romanov@nginx.com 
591347Smax.romanov@nginx.com     nxt_port_rpc_remove_peer(task, port, u.removed_pid);
592347Smax.romanov@nginx.com }
593347Smax.romanov@nginx.com 
594347Smax.romanov@nginx.com 
595192Smax.romanov@nginx.com void
596192Smax.romanov@nginx.com nxt_router_remove_pid_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
597192Smax.romanov@nginx.com {
598347Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
599318Smax.romanov@nginx.com 
600192Smax.romanov@nginx.com     nxt_port_remove_pid_handler(task, msg);
601192Smax.romanov@nginx.com 
602192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
603192Smax.romanov@nginx.com         return;
604192Smax.romanov@nginx.com     }
605192Smax.romanov@nginx.com 
606318Smax.romanov@nginx.com     nxt_queue_each(engine, &nxt_router->engines, nxt_event_engine_t, link0)
607318Smax.romanov@nginx.com     {
608347Smax.romanov@nginx.com         nxt_port_post(task, engine->port, nxt_router_worker_remove_pid,
609347Smax.romanov@nginx.com                       msg->u.data);
610318Smax.romanov@nginx.com     }
611318Smax.romanov@nginx.com     nxt_queue_loop;
612318Smax.romanov@nginx.com 
613192Smax.romanov@nginx.com     msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
614192Smax.romanov@nginx.com 
615192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
616192Smax.romanov@nginx.com }
617192Smax.romanov@nginx.com 
618192Smax.romanov@nginx.com 
61953Sigor@sysoev.ru static nxt_router_temp_conf_t *
620139Sigor@sysoev.ru nxt_router_temp_conf(nxt_task_t *task)
62153Sigor@sysoev.ru {
62265Sigor@sysoev.ru     nxt_mp_t                *mp, *tmp;
62353Sigor@sysoev.ru     nxt_router_conf_t       *rtcf;
62453Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
62553Sigor@sysoev.ru 
62665Sigor@sysoev.ru     mp = nxt_mp_create(1024, 128, 256, 32);
62753Sigor@sysoev.ru     if (nxt_slow_path(mp == NULL)) {
62853Sigor@sysoev.ru         return NULL;
62953Sigor@sysoev.ru     }
63053Sigor@sysoev.ru 
63165Sigor@sysoev.ru     rtcf = nxt_mp_zget(mp, sizeof(nxt_router_conf_t));
63253Sigor@sysoev.ru     if (nxt_slow_path(rtcf == NULL)) {
63353Sigor@sysoev.ru         goto fail;
63453Sigor@sysoev.ru     }
63553Sigor@sysoev.ru 
63653Sigor@sysoev.ru     rtcf->mem_pool = mp;
63753Sigor@sysoev.ru 
63865Sigor@sysoev.ru     tmp = nxt_mp_create(1024, 128, 256, 32);
63953Sigor@sysoev.ru     if (nxt_slow_path(tmp == NULL)) {
64053Sigor@sysoev.ru         goto fail;
64153Sigor@sysoev.ru     }
64253Sigor@sysoev.ru 
64365Sigor@sysoev.ru     tmcf = nxt_mp_zget(tmp, sizeof(nxt_router_temp_conf_t));
64453Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
64553Sigor@sysoev.ru         goto temp_fail;
64653Sigor@sysoev.ru     }
64753Sigor@sysoev.ru 
64853Sigor@sysoev.ru     tmcf->mem_pool = tmp;
64953Sigor@sysoev.ru     tmcf->conf = rtcf;
650139Sigor@sysoev.ru     tmcf->count = 1;
651139Sigor@sysoev.ru     tmcf->engine = task->thread->engine;
65253Sigor@sysoev.ru 
65353Sigor@sysoev.ru     tmcf->engines = nxt_array_create(tmcf->mem_pool, 4,
65453Sigor@sysoev.ru                                      sizeof(nxt_router_engine_conf_t));
65553Sigor@sysoev.ru     if (nxt_slow_path(tmcf->engines == NULL)) {
65653Sigor@sysoev.ru         goto temp_fail;
65753Sigor@sysoev.ru     }
65853Sigor@sysoev.ru 
65953Sigor@sysoev.ru     nxt_queue_init(&tmcf->deleting);
66053Sigor@sysoev.ru     nxt_queue_init(&tmcf->keeping);
66153Sigor@sysoev.ru     nxt_queue_init(&tmcf->updating);
66253Sigor@sysoev.ru     nxt_queue_init(&tmcf->pending);
66353Sigor@sysoev.ru     nxt_queue_init(&tmcf->creating);
664416Smax.romanov@nginx.com 
665133Sigor@sysoev.ru     nxt_queue_init(&tmcf->apps);
666133Sigor@sysoev.ru     nxt_queue_init(&tmcf->previous);
66753Sigor@sysoev.ru 
66853Sigor@sysoev.ru     return tmcf;
66953Sigor@sysoev.ru 
67053Sigor@sysoev.ru temp_fail:
67153Sigor@sysoev.ru 
67265Sigor@sysoev.ru     nxt_mp_destroy(tmp);
67353Sigor@sysoev.ru 
67453Sigor@sysoev.ru fail:
67553Sigor@sysoev.ru 
67665Sigor@sysoev.ru     nxt_mp_destroy(mp);
67753Sigor@sysoev.ru 
67853Sigor@sysoev.ru     return NULL;
67953Sigor@sysoev.ru }
68053Sigor@sysoev.ru 
68153Sigor@sysoev.ru 
682198Sigor@sysoev.ru static void
683198Sigor@sysoev.ru nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data)
684139Sigor@sysoev.ru {
685139Sigor@sysoev.ru     nxt_int_t                    ret;
686139Sigor@sysoev.ru     nxt_router_t                 *router;
687139Sigor@sysoev.ru     nxt_runtime_t                *rt;
688198Sigor@sysoev.ru     nxt_queue_link_t             *qlk;
689198Sigor@sysoev.ru     nxt_socket_conf_t            *skcf;
690198Sigor@sysoev.ru     nxt_router_temp_conf_t       *tmcf;
691139Sigor@sysoev.ru     const nxt_event_interface_t  *interface;
692139Sigor@sysoev.ru 
693198Sigor@sysoev.ru     tmcf = obj;
694198Sigor@sysoev.ru 
695198Sigor@sysoev.ru     qlk = nxt_queue_first(&tmcf->pending);
696198Sigor@sysoev.ru 
697198Sigor@sysoev.ru     if (qlk != nxt_queue_tail(&tmcf->pending)) {
698198Sigor@sysoev.ru         nxt_queue_remove(qlk);
699198Sigor@sysoev.ru         nxt_queue_insert_tail(&tmcf->creating, qlk);
700198Sigor@sysoev.ru 
701198Sigor@sysoev.ru         skcf = nxt_queue_link_data(qlk, nxt_socket_conf_t, link);
702198Sigor@sysoev.ru 
703198Sigor@sysoev.ru         nxt_router_listen_socket_rpc_create(task, tmcf, skcf);
704198Sigor@sysoev.ru 
705198Sigor@sysoev.ru         return;
706139Sigor@sysoev.ru     }
707139Sigor@sysoev.ru 
708139Sigor@sysoev.ru     rt = task->thread->runtime;
709139Sigor@sysoev.ru 
710139Sigor@sysoev.ru     interface = nxt_service_get(rt->services, "engine", NULL);
711139Sigor@sysoev.ru 
712198Sigor@sysoev.ru     router = tmcf->conf->router;
713198Sigor@sysoev.ru 
714139Sigor@sysoev.ru     ret = nxt_router_engines_create(task, router, tmcf, interface);
715139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
716198Sigor@sysoev.ru         goto fail;
717139Sigor@sysoev.ru     }
718139Sigor@sysoev.ru 
719139Sigor@sysoev.ru     ret = nxt_router_threads_create(task, rt, tmcf);
720139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
721198Sigor@sysoev.ru         goto fail;
722139Sigor@sysoev.ru     }
723139Sigor@sysoev.ru 
724343Smax.romanov@nginx.com     nxt_router_apps_sort(task, router, tmcf);
725139Sigor@sysoev.ru 
726315Sigor@sysoev.ru     nxt_router_engines_post(router, tmcf);
727139Sigor@sysoev.ru 
728139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->updating);
729139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->creating);
730139Sigor@sysoev.ru 
731198Sigor@sysoev.ru     nxt_router_conf_ready(task, tmcf);
732198Sigor@sysoev.ru 
733198Sigor@sysoev.ru     return;
734198Sigor@sysoev.ru 
735198Sigor@sysoev.ru fail:
736198Sigor@sysoev.ru 
737198Sigor@sysoev.ru     nxt_router_conf_error(task, tmcf);
738198Sigor@sysoev.ru 
739198Sigor@sysoev.ru     return;
740139Sigor@sysoev.ru }
741139Sigor@sysoev.ru 
742139Sigor@sysoev.ru 
743139Sigor@sysoev.ru static void
744139Sigor@sysoev.ru nxt_router_conf_wait(nxt_task_t *task, void *obj, void *data)
745139Sigor@sysoev.ru {
746153Sigor@sysoev.ru     nxt_joint_job_t  *job;
747153Sigor@sysoev.ru 
748153Sigor@sysoev.ru     job = obj;
749153Sigor@sysoev.ru 
750198Sigor@sysoev.ru     nxt_router_conf_ready(task, job->tmcf);
751139Sigor@sysoev.ru }
752139Sigor@sysoev.ru 
753139Sigor@sysoev.ru 
754139Sigor@sysoev.ru static void
755198Sigor@sysoev.ru nxt_router_conf_ready(nxt_task_t *task, nxt_router_temp_conf_t *tmcf)
756139Sigor@sysoev.ru {
757139Sigor@sysoev.ru     nxt_debug(task, "temp conf count:%D", tmcf->count);
758139Sigor@sysoev.ru 
759139Sigor@sysoev.ru     if (--tmcf->count == 0) {
760193Smax.romanov@nginx.com         nxt_router_conf_send(task, tmcf, NXT_PORT_MSG_RPC_READY_LAST);
761139Sigor@sysoev.ru     }
762139Sigor@sysoev.ru }
763139Sigor@sysoev.ru 
764139Sigor@sysoev.ru 
765139Sigor@sysoev.ru static void
766139Sigor@sysoev.ru nxt_router_conf_error(nxt_task_t *task, nxt_router_temp_conf_t *tmcf)
767139Sigor@sysoev.ru {
768148Sigor@sysoev.ru     nxt_socket_t       s;
769149Sigor@sysoev.ru     nxt_router_t       *router;
770148Sigor@sysoev.ru     nxt_queue_link_t   *qlk;
771148Sigor@sysoev.ru     nxt_socket_conf_t  *skcf;
772148Sigor@sysoev.ru 
773198Sigor@sysoev.ru     nxt_log(task, NXT_LOG_CRIT, "failed to apply new conf");
774198Sigor@sysoev.ru 
775148Sigor@sysoev.ru     for (qlk = nxt_queue_first(&tmcf->creating);
776148Sigor@sysoev.ru          qlk != nxt_queue_tail(&tmcf->creating);
777148Sigor@sysoev.ru          qlk = nxt_queue_next(qlk))
778148Sigor@sysoev.ru     {
779148Sigor@sysoev.ru         skcf = nxt_queue_link_data(qlk, nxt_socket_conf_t, link);
780359Sigor@sysoev.ru         s = skcf->listen->socket;
781148Sigor@sysoev.ru 
782148Sigor@sysoev.ru         if (s != -1) {
783148Sigor@sysoev.ru             nxt_socket_close(task, s);
784148Sigor@sysoev.ru         }
785148Sigor@sysoev.ru 
786359Sigor@sysoev.ru         nxt_free(skcf->listen);
787148Sigor@sysoev.ru     }
788148Sigor@sysoev.ru 
789149Sigor@sysoev.ru     router = tmcf->conf->router;
790149Sigor@sysoev.ru 
791149Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->keeping);
792149Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->deleting);
793149Sigor@sysoev.ru 
794416Smax.romanov@nginx.com     nxt_queue_add(&router->apps, &tmcf->previous);
795416Smax.romanov@nginx.com 
796148Sigor@sysoev.ru     // TODO: new engines and threads
797148Sigor@sysoev.ru 
798139Sigor@sysoev.ru     nxt_mp_destroy(tmcf->conf->mem_pool);
799139Sigor@sysoev.ru 
800193Smax.romanov@nginx.com     nxt_router_conf_send(task, tmcf, NXT_PORT_MSG_RPC_ERROR);
801139Sigor@sysoev.ru }
802139Sigor@sysoev.ru 
803139Sigor@sysoev.ru 
804139Sigor@sysoev.ru static void
805139Sigor@sysoev.ru nxt_router_conf_send(nxt_task_t *task, nxt_router_temp_conf_t *tmcf,
806193Smax.romanov@nginx.com     nxt_port_msg_type_t type)
807139Sigor@sysoev.ru {
808193Smax.romanov@nginx.com     nxt_port_socket_write(task, tmcf->port, type, -1, tmcf->stream, 0, NULL);
809139Sigor@sysoev.ru }
810139Sigor@sysoev.ru 
811139Sigor@sysoev.ru 
812115Sigor@sysoev.ru static nxt_conf_map_t  nxt_router_conf[] = {
813115Sigor@sysoev.ru     {
814133Sigor@sysoev.ru         nxt_string("listeners_threads"),
815115Sigor@sysoev.ru         NXT_CONF_MAP_INT32,
816115Sigor@sysoev.ru         offsetof(nxt_router_conf_t, threads),
817115Sigor@sysoev.ru     },
818115Sigor@sysoev.ru };
819115Sigor@sysoev.ru 
820115Sigor@sysoev.ru 
821133Sigor@sysoev.ru static nxt_conf_map_t  nxt_router_app_conf[] = {
822115Sigor@sysoev.ru     {
823133Sigor@sysoev.ru         nxt_string("type"),
824115Sigor@sysoev.ru         NXT_CONF_MAP_STR,
825133Sigor@sysoev.ru         offsetof(nxt_router_app_conf_t, type),
826115Sigor@sysoev.ru     },
827115Sigor@sysoev.ru 
828115Sigor@sysoev.ru     {
829133Sigor@sysoev.ru         nxt_string("workers"),
830115Sigor@sysoev.ru         NXT_CONF_MAP_INT32,
831133Sigor@sysoev.ru         offsetof(nxt_router_app_conf_t, workers),
832133Sigor@sysoev.ru     },
833318Smax.romanov@nginx.com 
834318Smax.romanov@nginx.com     {
835318Smax.romanov@nginx.com         nxt_string("limits"),
836318Smax.romanov@nginx.com         NXT_CONF_MAP_PTR,
837318Smax.romanov@nginx.com         offsetof(nxt_router_app_conf_t, limits_value),
838318Smax.romanov@nginx.com     },
839318Smax.romanov@nginx.com };
840318Smax.romanov@nginx.com 
841318Smax.romanov@nginx.com 
842318Smax.romanov@nginx.com static nxt_conf_map_t  nxt_router_app_limits_conf[] = {
843318Smax.romanov@nginx.com     {
844318Smax.romanov@nginx.com         nxt_string("timeout"),
845318Smax.romanov@nginx.com         NXT_CONF_MAP_MSEC,
846318Smax.romanov@nginx.com         offsetof(nxt_router_app_conf_t, timeout),
847318Smax.romanov@nginx.com     },
848318Smax.romanov@nginx.com 
849318Smax.romanov@nginx.com     {
850318Smax.romanov@nginx.com         nxt_string("requests"),
851318Smax.romanov@nginx.com         NXT_CONF_MAP_INT32,
852318Smax.romanov@nginx.com         offsetof(nxt_router_app_conf_t, requests),
853318Smax.romanov@nginx.com     },
854133Sigor@sysoev.ru };
855133Sigor@sysoev.ru 
856133Sigor@sysoev.ru 
857133Sigor@sysoev.ru static nxt_conf_map_t  nxt_router_listener_conf[] = {
858133Sigor@sysoev.ru     {
859133Sigor@sysoev.ru         nxt_string("application"),
860133Sigor@sysoev.ru         NXT_CONF_MAP_STR,
861133Sigor@sysoev.ru         offsetof(nxt_router_listener_conf_t, application),
862115Sigor@sysoev.ru     },
863115Sigor@sysoev.ru };
864115Sigor@sysoev.ru 
865115Sigor@sysoev.ru 
866115Sigor@sysoev.ru static nxt_conf_map_t  nxt_router_http_conf[] = {
867115Sigor@sysoev.ru     {
868115Sigor@sysoev.ru         nxt_string("header_buffer_size"),
869115Sigor@sysoev.ru         NXT_CONF_MAP_SIZE,
870115Sigor@sysoev.ru         offsetof(nxt_socket_conf_t, header_buffer_size),
871