xref: /unit/src/nxt_router.c (revision 345)
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 
26167Smax.romanov@nginx.com typedef struct nxt_req_app_link_s nxt_req_app_link_t;
27141Smax.romanov@nginx.com 
28141Smax.romanov@nginx.com 
29318Smax.romanov@nginx.com typedef struct {
30318Smax.romanov@nginx.com     uint32_t             stream;
31318Smax.romanov@nginx.com     nxt_conn_t           *conn;
32343Smax.romanov@nginx.com     nxt_app_t            *app;
33318Smax.romanov@nginx.com     nxt_port_t           *app_port;
34318Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
35318Smax.romanov@nginx.com 
36318Smax.romanov@nginx.com     nxt_queue_link_t     link;     /* for nxt_conn_t.requests */
37318Smax.romanov@nginx.com } nxt_req_conn_link_t;
38318Smax.romanov@nginx.com 
39318Smax.romanov@nginx.com 
40167Smax.romanov@nginx.com struct nxt_req_app_link_s {
41318Smax.romanov@nginx.com     uint32_t             stream;
42167Smax.romanov@nginx.com     nxt_port_t           *app_port;
43318Smax.romanov@nginx.com     nxt_pid_t            app_pid;
44167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
45167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
46167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
47167Smax.romanov@nginx.com 
48167Smax.romanov@nginx.com     nxt_queue_link_t     link; /* for nxt_app_t.requests */
49167Smax.romanov@nginx.com 
50167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
51167Smax.romanov@nginx.com     nxt_work_t           work;
52*345Smax.romanov@nginx.com 
53*345Smax.romanov@nginx.com     int                  err_code;
54*345Smax.romanov@nginx.com     const char           *err_str;
55167Smax.romanov@nginx.com };
56167Smax.romanov@nginx.com 
57167Smax.romanov@nginx.com 
58198Sigor@sysoev.ru typedef struct {
59198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
60198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
61198Sigor@sysoev.ru } nxt_socket_rpc_t;
62198Sigor@sysoev.ru 
63198Sigor@sysoev.ru 
64318Smax.romanov@nginx.com typedef struct {
65318Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
66318Smax.romanov@nginx.com     nxt_port_recv_msg_t  msg;
67318Smax.romanov@nginx.com     nxt_work_t           work;
68318Smax.romanov@nginx.com } nxt_remove_pid_msg_t;
69318Smax.romanov@nginx.com 
70318Smax.romanov@nginx.com 
71343Smax.romanov@nginx.com static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app);
72343Smax.romanov@nginx.com 
73*345Smax.romanov@nginx.com static void nxt_router_ra_error(nxt_task_t *task, nxt_req_app_link_t *ra,
74*345Smax.romanov@nginx.com     int code, const char* str);
75*345Smax.romanov@nginx.com 
76318Smax.romanov@nginx.com static void nxt_router_worker_remove_pid_handler(nxt_task_t *task, void *obj,
77318Smax.romanov@nginx.com     void *data);
78318Smax.romanov@nginx.com static void nxt_router_worker_remove_pid_done(nxt_task_t *task, void *obj,
79318Smax.romanov@nginx.com     void *data);
80318Smax.romanov@nginx.com 
81139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
82198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
83198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
84139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
85139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
86139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
87139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
88193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
8953Sigor@sysoev.ru static void nxt_router_listen_sockets_sort(nxt_router_t *router,
9053Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
9153Sigor@sysoev.ru 
92115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
93115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
94133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
95133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
96133Sigor@sysoev.ru     nxt_str_t *name);
97198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
98198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
99198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
100198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
101198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
102198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
10365Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task, nxt_mp_t *mp,
10465Sigor@sysoev.ru     nxt_sockaddr_t *sa);
10553Sigor@sysoev.ru 
10653Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
10753Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
10853Sigor@sysoev.ru     const nxt_event_interface_t *interface);
109115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
110115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
111115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
112115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
113115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
114115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
115115Sigor@sysoev.ru static void nxt_router_engine_socket_count(nxt_queue_t *sockets);
116154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
117154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
118154Sigor@sysoev.ru     nxt_work_handler_t handler);
119313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
120313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
121139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
122139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
12353Sigor@sysoev.ru 
12453Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
12553Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
12653Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
12753Sigor@sysoev.ru     nxt_event_engine_t *engine);
128343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
129133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
13053Sigor@sysoev.ru 
131315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
132315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
133315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
134315Sigor@sysoev.ru     nxt_work_t *jobs);
13553Sigor@sysoev.ru 
13653Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
13753Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
13853Sigor@sysoev.ru     void *data);
13953Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
14053Sigor@sysoev.ru     void *data);
14153Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
14253Sigor@sysoev.ru     void *data);
143313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
144313Sigor@sysoev.ru     void *data);
14553Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
14653Sigor@sysoev.ru     void *data);
14753Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
14853Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
14953Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
15053Sigor@sysoev.ru     void *data);
15153Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
15253Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
15353Sigor@sysoev.ru 
154343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
155343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
156343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
157343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
158343Smax.romanov@nginx.com 
159343Smax.romanov@nginx.com static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app);
160343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
161343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
162141Smax.romanov@nginx.com 
16353Sigor@sysoev.ru static void nxt_router_conn_init(nxt_task_t *task, void *obj, void *data);
16453Sigor@sysoev.ru static void nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj,
16553Sigor@sysoev.ru     void *data);
166206Smax.romanov@nginx.com static void nxt_router_conn_http_body_read(nxt_task_t *task, void *obj,
167206Smax.romanov@nginx.com     void *data);
16888Smax.romanov@nginx.com static void nxt_router_process_http_request(nxt_task_t *task,
16988Smax.romanov@nginx.com     nxt_conn_t *c, nxt_app_parse_ctx_t *ap);
170141Smax.romanov@nginx.com static void nxt_router_process_http_request_mp(nxt_task_t *task,
171343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
172216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
173216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
174216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
175216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
176216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
177216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
17888Smax.romanov@nginx.com static void nxt_router_conn_ready(nxt_task_t *task, void *obj, void *data);
17953Sigor@sysoev.ru static void nxt_router_conn_close(nxt_task_t *task, void *obj, void *data);
18053Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
18153Sigor@sysoev.ru static void nxt_router_conn_error(nxt_task_t *task, void *obj, void *data);
18253Sigor@sysoev.ru static void nxt_router_conn_timeout(nxt_task_t *task, void *obj, void *data);
183318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
18462Sigor@sysoev.ru static nxt_msec_t nxt_router_conn_timeout_value(nxt_conn_t *c, uintptr_t data);
18520Sigor@sysoev.ru 
186141Smax.romanov@nginx.com static void nxt_router_gen_error(nxt_task_t *task, nxt_conn_t *c, int code,
187*345Smax.romanov@nginx.com     const char* str);
188141Smax.romanov@nginx.com 
189119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
19020Sigor@sysoev.ru 
191216Sigor@sysoev.ru 
192216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
193216Sigor@sysoev.ru     nxt_python_prepare_msg,
194216Sigor@sysoev.ru     nxt_php_prepare_msg,
195216Sigor@sysoev.ru     nxt_go_prepare_msg,
196216Sigor@sysoev.ru };
197216Sigor@sysoev.ru 
198216Sigor@sysoev.ru 
19920Sigor@sysoev.ru nxt_int_t
200141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
20120Sigor@sysoev.ru {
202141Smax.romanov@nginx.com     nxt_int_t      ret;
203141Smax.romanov@nginx.com     nxt_router_t   *router;
204141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
205141Smax.romanov@nginx.com 
206141Smax.romanov@nginx.com     rt = task->thread->runtime;
20753Sigor@sysoev.ru 
20888Smax.romanov@nginx.com     ret = nxt_app_http_init(task, rt);
20988Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
21088Smax.romanov@nginx.com         return ret;
21188Smax.romanov@nginx.com     }
21288Smax.romanov@nginx.com 
21353Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
21453Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
21553Sigor@sysoev.ru         return NXT_ERROR;
21653Sigor@sysoev.ru     }
21753Sigor@sysoev.ru 
21853Sigor@sysoev.ru     nxt_queue_init(&router->engines);
21953Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
220133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
22153Sigor@sysoev.ru 
222119Smax.romanov@nginx.com     nxt_router = router;
223119Smax.romanov@nginx.com 
224115Sigor@sysoev.ru     return NXT_OK;
225115Sigor@sysoev.ru }
226115Sigor@sysoev.ru 
227115Sigor@sysoev.ru 
228343Smax.romanov@nginx.com static void
229343Smax.romanov@nginx.com nxt_router_start_worker_handler(nxt_task_t *task, nxt_port_t *port, void *data)
230167Smax.romanov@nginx.com {
231343Smax.romanov@nginx.com     size_t         size;
232343Smax.romanov@nginx.com     uint32_t       stream;
233343Smax.romanov@nginx.com     nxt_app_t      *app;
234343Smax.romanov@nginx.com     nxt_buf_t      *b;
235343Smax.romanov@nginx.com     nxt_port_t     *main_port;
236343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
237343Smax.romanov@nginx.com 
238343Smax.romanov@nginx.com     app = data;
239167Smax.romanov@nginx.com 
240167Smax.romanov@nginx.com     rt = task->thread->runtime;
241240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
242167Smax.romanov@nginx.com 
243343Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start worker", &app->name, app);
244343Smax.romanov@nginx.com 
245343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
246343Smax.romanov@nginx.com 
247343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
248343Smax.romanov@nginx.com 
249343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
250343Smax.romanov@nginx.com         goto failed;
251167Smax.romanov@nginx.com     }
252167Smax.romanov@nginx.com 
253343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
254343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
255343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
256343Smax.romanov@nginx.com 
257343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
258343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
259343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
260343Smax.romanov@nginx.com                                            -1, app);
261343Smax.romanov@nginx.com 
262343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
263343Smax.romanov@nginx.com         nxt_mp_release(b->data, b);
264343Smax.romanov@nginx.com 
265343Smax.romanov@nginx.com         goto failed;
266343Smax.romanov@nginx.com     }
267343Smax.romanov@nginx.com 
268343Smax.romanov@nginx.com     nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
269343Smax.romanov@nginx.com                           stream, port->id, b);
270343Smax.romanov@nginx.com 
271343Smax.romanov@nginx.com     return;
272343Smax.romanov@nginx.com 
273343Smax.romanov@nginx.com failed:
274343Smax.romanov@nginx.com 
275343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
276343Smax.romanov@nginx.com 
277343Smax.romanov@nginx.com     app->pending_workers--;
278343Smax.romanov@nginx.com 
279343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
280343Smax.romanov@nginx.com 
281343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
282167Smax.romanov@nginx.com }
283167Smax.romanov@nginx.com 
284167Smax.romanov@nginx.com 
285343Smax.romanov@nginx.com static nxt_int_t
286343Smax.romanov@nginx.com nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app)
287141Smax.romanov@nginx.com {
288343Smax.romanov@nginx.com     nxt_int_t      res;
289343Smax.romanov@nginx.com     nxt_port_t     *router_port;
290343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
291343Smax.romanov@nginx.com 
292343Smax.romanov@nginx.com     rt = task->thread->runtime;
293343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
294343Smax.romanov@nginx.com 
295343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
296343Smax.romanov@nginx.com 
297343Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_worker_handler,
298343Smax.romanov@nginx.com                         app);
299343Smax.romanov@nginx.com 
300343Smax.romanov@nginx.com     if (res == NXT_OK) {
301343Smax.romanov@nginx.com         return res;
302318Smax.romanov@nginx.com     }
303318Smax.romanov@nginx.com 
304343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
305343Smax.romanov@nginx.com 
306343Smax.romanov@nginx.com     app->pending_workers--;
307343Smax.romanov@nginx.com 
308343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
309343Smax.romanov@nginx.com 
310343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
311343Smax.romanov@nginx.com 
312343Smax.romanov@nginx.com     return NXT_ERROR;
313318Smax.romanov@nginx.com }
314318Smax.romanov@nginx.com 
315318Smax.romanov@nginx.com 
316167Smax.romanov@nginx.com static nxt_req_app_link_t *
317167Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_conn_link_t *rc)
318167Smax.romanov@nginx.com {
319167Smax.romanov@nginx.com     nxt_mp_t            *mp;
320318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
321167Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
322167Smax.romanov@nginx.com 
323167Smax.romanov@nginx.com     mp = rc->conn->mem_pool;
324318Smax.romanov@nginx.com     engine = task->thread->engine;
325167Smax.romanov@nginx.com 
326167Smax.romanov@nginx.com     ra = nxt_mp_retain(mp, sizeof(nxt_req_app_link_t));
327167Smax.romanov@nginx.com 
328167Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
329167Smax.romanov@nginx.com         return NULL;
330167Smax.romanov@nginx.com     }
331167Smax.romanov@nginx.com 
332318Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD create", rc->stream);
333167Smax.romanov@nginx.com 
334167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
335167Smax.romanov@nginx.com 
336318Smax.romanov@nginx.com     ra->stream = rc->stream;
337318Smax.romanov@nginx.com     ra->app_pid = -1;
338167Smax.romanov@nginx.com     ra->rc = rc;
339318Smax.romanov@nginx.com     rc->ra = ra;
340318Smax.romanov@nginx.com     ra->reply_port = engine->port;
341167Smax.romanov@nginx.com 
342167Smax.romanov@nginx.com     ra->mem_pool = mp;
343167Smax.romanov@nginx.com 
344167Smax.romanov@nginx.com     ra->work.handler = NULL;
345318Smax.romanov@nginx.com     ra->work.task = &engine->task;
346167Smax.romanov@nginx.com     ra->work.obj = ra;
347318Smax.romanov@nginx.com     ra->work.data = engine;
348167Smax.romanov@nginx.com 
349167Smax.romanov@nginx.com     return ra;
350167Smax.romanov@nginx.com }
351167Smax.romanov@nginx.com 
352167Smax.romanov@nginx.com 
353167Smax.romanov@nginx.com static void
354167Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, void *obj, void *data)
355167Smax.romanov@nginx.com {
356343Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
357343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
358343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
359318Smax.romanov@nginx.com 
360318Smax.romanov@nginx.com     ra = obj;
361318Smax.romanov@nginx.com     engine = data;
362318Smax.romanov@nginx.com 
363343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
364343Smax.romanov@nginx.com         if (ra->app_port != NULL) {
365343Smax.romanov@nginx.com             ra->app_pid = ra->app_port->pid;
366318Smax.romanov@nginx.com         }
367318Smax.romanov@nginx.com 
368318Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release;
369318Smax.romanov@nginx.com         ra->work.task = &engine->task;
370318Smax.romanov@nginx.com         ra->work.next = NULL;
371318Smax.romanov@nginx.com 
372318Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
373318Smax.romanov@nginx.com                   ra->stream, engine);
374318Smax.romanov@nginx.com 
375318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
376318Smax.romanov@nginx.com 
377318Smax.romanov@nginx.com         return;
378318Smax.romanov@nginx.com     }
379318Smax.romanov@nginx.com 
380343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
381343Smax.romanov@nginx.com 
382343Smax.romanov@nginx.com     rc = ra->rc;
383343Smax.romanov@nginx.com 
384343Smax.romanov@nginx.com     if (rc != NULL) {
385343Smax.romanov@nginx.com         if (ra->app_pid != -1) {
386343Smax.romanov@nginx.com             nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_pid);
387343Smax.romanov@nginx.com         }
388343Smax.romanov@nginx.com 
389343Smax.romanov@nginx.com         rc->app_port = ra->app_port;
390343Smax.romanov@nginx.com 
391343Smax.romanov@nginx.com         ra->app_port = NULL;
392343Smax.romanov@nginx.com         rc->ra = NULL;
393343Smax.romanov@nginx.com         ra->rc = NULL;
394318Smax.romanov@nginx.com     }
395318Smax.romanov@nginx.com 
396343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
397343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
398343Smax.romanov@nginx.com 
399343Smax.romanov@nginx.com         ra->app_port = NULL;
400343Smax.romanov@nginx.com     }
401318Smax.romanov@nginx.com 
402318Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
403318Smax.romanov@nginx.com }
404318Smax.romanov@nginx.com 
405318Smax.romanov@nginx.com 
406318Smax.romanov@nginx.com static void
407318Smax.romanov@nginx.com nxt_router_ra_abort(nxt_task_t *task, void *obj, void *data)
408318Smax.romanov@nginx.com {
409343Smax.romanov@nginx.com     nxt_conn_t           *c;
410343Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
411343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
412343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
413167Smax.romanov@nginx.com 
414167Smax.romanov@nginx.com     ra = obj;
415167Smax.romanov@nginx.com     engine = data;
416167Smax.romanov@nginx.com 
417167Smax.romanov@nginx.com     if (task->thread->engine != engine) {
418318Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_abort;
419167Smax.romanov@nginx.com         ra->work.task = &engine->task;
420167Smax.romanov@nginx.com         ra->work.next = NULL;
421167Smax.romanov@nginx.com 
422318Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post abort to %p", ra->stream, engine);
423167Smax.romanov@nginx.com 
424167Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
425167Smax.romanov@nginx.com 
426167Smax.romanov@nginx.com         return;
427167Smax.romanov@nginx.com     }
428167Smax.romanov@nginx.com 
429318Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD abort", ra->stream);
430318Smax.romanov@nginx.com 
431343Smax.romanov@nginx.com     rc = ra->rc;
432343Smax.romanov@nginx.com 
433343Smax.romanov@nginx.com     if (rc != NULL) {
434343Smax.romanov@nginx.com         c = rc->conn;
435318Smax.romanov@nginx.com 
436318Smax.romanov@nginx.com         nxt_router_gen_error(task, c, 500,
437318Smax.romanov@nginx.com                              "Failed to start application worker");
438343Smax.romanov@nginx.com 
439343Smax.romanov@nginx.com         rc->ra = NULL;
440343Smax.romanov@nginx.com         ra->rc = NULL;
441343Smax.romanov@nginx.com     }
442343Smax.romanov@nginx.com 
443343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
444343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
445343Smax.romanov@nginx.com 
446343Smax.romanov@nginx.com         ra->app_port = NULL;
447167Smax.romanov@nginx.com     }
448167Smax.romanov@nginx.com 
449167Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
450167Smax.romanov@nginx.com }
451167Smax.romanov@nginx.com 
452167Smax.romanov@nginx.com 
453*345Smax.romanov@nginx.com static void
454*345Smax.romanov@nginx.com nxt_router_ra_error_handler(nxt_task_t *task, void *obj, void *data)
455*345Smax.romanov@nginx.com {
456*345Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
457*345Smax.romanov@nginx.com 
458*345Smax.romanov@nginx.com     ra = obj;
459*345Smax.romanov@nginx.com 
460*345Smax.romanov@nginx.com     nxt_router_ra_error(task, ra, ra->err_code, ra->err_str);
461*345Smax.romanov@nginx.com }
462*345Smax.romanov@nginx.com 
463*345Smax.romanov@nginx.com 
464*345Smax.romanov@nginx.com static void
465*345Smax.romanov@nginx.com nxt_router_ra_error(nxt_task_t *task, nxt_req_app_link_t *ra, int code,
466*345Smax.romanov@nginx.com     const char* str)
467*345Smax.romanov@nginx.com {
468*345Smax.romanov@nginx.com     nxt_conn_t           *c;
469*345Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
470*345Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
471*345Smax.romanov@nginx.com 
472*345Smax.romanov@nginx.com     engine = ra->work.data;
473*345Smax.romanov@nginx.com 
474*345Smax.romanov@nginx.com     if (task->thread->engine != engine) {
475*345Smax.romanov@nginx.com         ra->err_code = code;
476*345Smax.romanov@nginx.com         ra->err_str = str;
477*345Smax.romanov@nginx.com 
478*345Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_error_handler;
479*345Smax.romanov@nginx.com         ra->work.task = &engine->task;
480*345Smax.romanov@nginx.com         ra->work.next = NULL;
481*345Smax.romanov@nginx.com 
482*345Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post error to %p", ra->stream, engine);
483*345Smax.romanov@nginx.com 
484*345Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
485*345Smax.romanov@nginx.com 
486*345Smax.romanov@nginx.com         return;
487*345Smax.romanov@nginx.com     }
488*345Smax.romanov@nginx.com 
489*345Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD error", ra->stream);
490*345Smax.romanov@nginx.com 
491*345Smax.romanov@nginx.com     rc = ra->rc;
492*345Smax.romanov@nginx.com 
493*345Smax.romanov@nginx.com     if (rc != NULL) {
494*345Smax.romanov@nginx.com         c = rc->conn;
495*345Smax.romanov@nginx.com 
496*345Smax.romanov@nginx.com         nxt_router_gen_error(task, c, code, str);
497*345Smax.romanov@nginx.com 
498*345Smax.romanov@nginx.com         rc->ra = NULL;
499*345Smax.romanov@nginx.com         ra->rc = NULL;
500*345Smax.romanov@nginx.com     }
501*345Smax.romanov@nginx.com 
502*345Smax.romanov@nginx.com     if (ra->app_port != NULL) {
503*345Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
504*345Smax.romanov@nginx.com 
505*345Smax.romanov@nginx.com         ra->app_port = NULL;
506*345Smax.romanov@nginx.com     }
507*345Smax.romanov@nginx.com 
508*345Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
509*345Smax.romanov@nginx.com }
510*345Smax.romanov@nginx.com 
511*345Smax.romanov@nginx.com 
512343Smax.romanov@nginx.com nxt_inline void
513343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
514343Smax.romanov@nginx.com {
515343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
516343Smax.romanov@nginx.com 
517343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
518343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
519343Smax.romanov@nginx.com 
520343Smax.romanov@nginx.com         rc->app_port = NULL;
521343Smax.romanov@nginx.com     }
522343Smax.romanov@nginx.com 
523343Smax.romanov@nginx.com     ra = rc->ra;
524343Smax.romanov@nginx.com 
525343Smax.romanov@nginx.com     if (ra != NULL) {
526343Smax.romanov@nginx.com         rc->ra = NULL;
527343Smax.romanov@nginx.com         ra->rc = NULL;
528343Smax.romanov@nginx.com 
529343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
530343Smax.romanov@nginx.com 
531343Smax.romanov@nginx.com         if (ra->link.next != NULL) {
532343Smax.romanov@nginx.com             nxt_queue_remove(&ra->link);
533343Smax.romanov@nginx.com 
534343Smax.romanov@nginx.com             ra->link.next = NULL;
535343Smax.romanov@nginx.com 
536343Smax.romanov@nginx.com         } else {
537343Smax.romanov@nginx.com             ra = NULL;
538343Smax.romanov@nginx.com         }
539343Smax.romanov@nginx.com 
540343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
541343Smax.romanov@nginx.com     }
542343Smax.romanov@nginx.com 
543343Smax.romanov@nginx.com     if (ra != NULL) {
544343Smax.romanov@nginx.com         nxt_router_ra_release(task, ra, ra->work.data);
545343Smax.romanov@nginx.com     }
546343Smax.romanov@nginx.com 
547343Smax.romanov@nginx.com     if (rc->app != NULL) {
548343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
549343Smax.romanov@nginx.com 
550343Smax.romanov@nginx.com         rc->app = NULL;
551343Smax.romanov@nginx.com     }
552343Smax.romanov@nginx.com 
553343Smax.romanov@nginx.com     nxt_queue_remove(&rc->link);
554343Smax.romanov@nginx.com 
555343Smax.romanov@nginx.com     rc->conn = NULL;
556343Smax.romanov@nginx.com }
557343Smax.romanov@nginx.com 
558343Smax.romanov@nginx.com 
559141Smax.romanov@nginx.com void
560141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
561141Smax.romanov@nginx.com {
562141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
563141Smax.romanov@nginx.com 
564192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
565141Smax.romanov@nginx.com         return;
566141Smax.romanov@nginx.com     }
567141Smax.romanov@nginx.com 
568192Smax.romanov@nginx.com     if (msg->new_port == NULL || msg->new_port->type != NXT_PROCESS_WORKER) {
569192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
570141Smax.romanov@nginx.com     }
571192Smax.romanov@nginx.com 
572192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
573141Smax.romanov@nginx.com }
574141Smax.romanov@nginx.com 
575141Smax.romanov@nginx.com 
576139Sigor@sysoev.ru void
577139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
578115Sigor@sysoev.ru {
579139Sigor@sysoev.ru     size_t                  dump_size;
580198Sigor@sysoev.ru     nxt_int_t               ret;
581139Sigor@sysoev.ru     nxt_buf_t               *b;
582139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
583139Sigor@sysoev.ru 
584139Sigor@sysoev.ru     b = msg->buf;
585139Sigor@sysoev.ru 
586139Sigor@sysoev.ru     dump_size = nxt_buf_used_size(b);
587139Sigor@sysoev.ru 
588139Sigor@sysoev.ru     if (dump_size > 300) {
589139Sigor@sysoev.ru         dump_size = 300;
59053Sigor@sysoev.ru     }
59153Sigor@sysoev.ru 
592139Sigor@sysoev.ru     nxt_debug(task, "router conf data (%z): %*s",
593139Sigor@sysoev.ru               msg->size, dump_size, b->mem.pos);
594139Sigor@sysoev.ru 
595139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
596139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
597139Sigor@sysoev.ru         return;
59853Sigor@sysoev.ru     }
59953Sigor@sysoev.ru 
600139Sigor@sysoev.ru     tmcf->conf->router = nxt_router;
601139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
602139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
603198Sigor@sysoev.ru                                        msg->port_msg.pid,
604198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
605198Sigor@sysoev.ru 
606198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
607198Sigor@sysoev.ru 
608198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
609198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
610198Sigor@sysoev.ru 
611198Sigor@sysoev.ru     } else {
612198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
613139Sigor@sysoev.ru     }
61453Sigor@sysoev.ru }
61553Sigor@sysoev.ru 
61653Sigor@sysoev.ru 
617192Smax.romanov@nginx.com void
618192Smax.romanov@nginx.com nxt_router_remove_pid_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
619192Smax.romanov@nginx.com {
620318Smax.romanov@nginx.com     nxt_mp_t              *mp;
621318Smax.romanov@nginx.com     nxt_buf_t             *buf;
622318Smax.romanov@nginx.com     nxt_event_engine_t    *engine;
623318Smax.romanov@nginx.com     nxt_remove_pid_msg_t  *rp;
624318Smax.romanov@nginx.com 
625192Smax.romanov@nginx.com     nxt_port_remove_pid_handler(task, msg);
626192Smax.romanov@nginx.com 
627192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
628192Smax.romanov@nginx.com         return;
629192Smax.romanov@nginx.com     }
630192Smax.romanov@nginx.com 
631318Smax.romanov@nginx.com     mp = nxt_mp_create(1024, 128, 256, 32);
632318Smax.romanov@nginx.com 
633318Smax.romanov@nginx.com     buf = nxt_buf_mem_alloc(mp, nxt_buf_used_size(msg->buf), 0);
634318Smax.romanov@nginx.com     buf->mem.free = nxt_cpymem(buf->mem.free, msg->buf->mem.pos,
635318Smax.romanov@nginx.com                                nxt_buf_used_size(msg->buf));
636318Smax.romanov@nginx.com 
637318Smax.romanov@nginx.com     nxt_queue_each(engine, &nxt_router->engines, nxt_event_engine_t, link0)
638318Smax.romanov@nginx.com     {
639318Smax.romanov@nginx.com         rp = nxt_mp_retain(mp, sizeof(nxt_remove_pid_msg_t));
640318Smax.romanov@nginx.com 
641318Smax.romanov@nginx.com         rp->mem_pool = mp;
642318Smax.romanov@nginx.com 
643318Smax.romanov@nginx.com         rp->msg.fd = msg->fd;
644318Smax.romanov@nginx.com         rp->msg.buf = buf;
645318Smax.romanov@nginx.com         rp->msg.port = engine->port;
646318Smax.romanov@nginx.com         rp->msg.port_msg = msg->port_msg;
647318Smax.romanov@nginx.com         rp->msg.size = msg->size;
648318Smax.romanov@nginx.com         rp->msg.new_port = NULL;
649318Smax.romanov@nginx.com 
650318Smax.romanov@nginx.com         rp->work.handler = nxt_router_worker_remove_pid_handler;
651318Smax.romanov@nginx.com         rp->work.task = &engine->task;
652318Smax.romanov@nginx.com         rp->work.obj = rp;
653318Smax.romanov@nginx.com         rp->work.data = task->thread->engine;
654318Smax.romanov@nginx.com         rp->work.next = NULL;
655318Smax.romanov@nginx.com 
656318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &rp->work);
657318Smax.romanov@nginx.com     }
658318Smax.romanov@nginx.com     nxt_queue_loop;
659318Smax.romanov@nginx.com 
660340Smax.romanov@nginx.com     nxt_mp_release(mp, NULL);
661340Smax.romanov@nginx.com 
662192Smax.romanov@nginx.com     msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
663192Smax.romanov@nginx.com 
664192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
665192Smax.romanov@nginx.com }
666192Smax.romanov@nginx.com 
667192Smax.romanov@nginx.com 
668318Smax.romanov@nginx.com static void
669318Smax.romanov@nginx.com nxt_router_worker_remove_pid_handler(nxt_task_t *task, void *obj, void *data)
670318Smax.romanov@nginx.com {
671343Smax.romanov@nginx.com     nxt_pid_t             pid;
672343Smax.romanov@nginx.com     nxt_buf_t             *buf;
673318Smax.romanov@nginx.com     nxt_event_engine_t    *engine;
674318Smax.romanov@nginx.com     nxt_remove_pid_msg_t  *rp;
675318Smax.romanov@nginx.com 
676318Smax.romanov@nginx.com     rp = obj;
677318Smax.romanov@nginx.com 
678343Smax.romanov@nginx.com     buf = rp->msg.buf;
679343Smax.romanov@nginx.com 
680343Smax.romanov@nginx.com     nxt_assert(nxt_buf_used_size(buf) == sizeof(pid));
681343Smax.romanov@nginx.com 
682343Smax.romanov@nginx.com     nxt_memcpy(&pid, buf->mem.pos, sizeof(pid));
683343Smax.romanov@nginx.com 
684343Smax.romanov@nginx.com     nxt_port_rpc_remove_peer(task, rp->msg.port, pid);
685318Smax.romanov@nginx.com 
686318Smax.romanov@nginx.com     engine = rp->work.data;
687318Smax.romanov@nginx.com 
688318Smax.romanov@nginx.com     rp->work.handler = nxt_router_worker_remove_pid_done;
689318Smax.romanov@nginx.com     rp->work.task = &engine->task;
690318Smax.romanov@nginx.com     rp->work.next = NULL;
691318Smax.romanov@nginx.com 
692318Smax.romanov@nginx.com     nxt_event_engine_post(engine, &rp->work);
693318Smax.romanov@nginx.com }
694318Smax.romanov@nginx.com 
695318Smax.romanov@nginx.com 
696318Smax.romanov@nginx.com static void
697318Smax.romanov@nginx.com nxt_router_worker_remove_pid_done(nxt_task_t *task, void *obj, void *data)
698318Smax.romanov@nginx.com {
699318Smax.romanov@nginx.com     nxt_remove_pid_msg_t  *rp;
700318Smax.romanov@nginx.com 
701318Smax.romanov@nginx.com     rp = obj;
702318Smax.romanov@nginx.com 
703318Smax.romanov@nginx.com     nxt_mp_release(rp->mem_pool, rp);
704318Smax.romanov@nginx.com }
705318Smax.romanov@nginx.com 
706318Smax.romanov@nginx.com 
70753Sigor@sysoev.ru static nxt_router_temp_conf_t *
708139Sigor@sysoev.ru nxt_router_temp_conf(nxt_task_t *task)
70953Sigor@sysoev.ru {
71065Sigor@sysoev.ru     nxt_mp_t                *mp, *tmp;
71153Sigor@sysoev.ru     nxt_router_conf_t       *rtcf;
71253Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
71353Sigor@sysoev.ru 
71465Sigor@sysoev.ru     mp = nxt_mp_create(1024, 128, 256, 32);
71553Sigor@sysoev.ru     if (nxt_slow_path(mp == NULL)) {
71653Sigor@sysoev.ru         return NULL;
71753Sigor@sysoev.ru     }
71853Sigor@sysoev.ru 
71965Sigor@sysoev.ru     rtcf = nxt_mp_zget(mp, sizeof(nxt_router_conf_t));
72053Sigor@sysoev.ru     if (nxt_slow_path(rtcf == NULL)) {
72153Sigor@sysoev.ru         goto fail;
72253Sigor@sysoev.ru     }
72353Sigor@sysoev.ru 
72453Sigor@sysoev.ru     rtcf->mem_pool = mp;
72553Sigor@sysoev.ru 
72665Sigor@sysoev.ru     tmp = nxt_mp_create(1024, 128, 256, 32);
72753Sigor@sysoev.ru     if (nxt_slow_path(tmp == NULL)) {
72853Sigor@sysoev.ru         goto fail;
72953Sigor@sysoev.ru     }
73053Sigor@sysoev.ru 
73165Sigor@sysoev.ru     tmcf = nxt_mp_zget(tmp, sizeof(nxt_router_temp_conf_t));
73253Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
73353Sigor@sysoev.ru         goto temp_fail;
73453Sigor@sysoev.ru     }
73553Sigor@sysoev.ru 
73653Sigor@sysoev.ru     tmcf->mem_pool = tmp;
73753Sigor@sysoev.ru     tmcf->conf = rtcf;
738139Sigor@sysoev.ru     tmcf->count = 1;
739139Sigor@sysoev.ru     tmcf->engine = task->thread->engine;
74053Sigor@sysoev.ru 
74153Sigor@sysoev.ru     tmcf->engines = nxt_array_create(tmcf->mem_pool, 4,
74253Sigor@sysoev.ru                                      sizeof(nxt_router_engine_conf_t));
74353Sigor@sysoev.ru     if (nxt_slow_path(tmcf->engines == NULL)) {
74453Sigor@sysoev.ru         goto temp_fail;
74553Sigor@sysoev.ru     }
74653Sigor@sysoev.ru 
74753Sigor@sysoev.ru     nxt_queue_init(&tmcf->deleting);
74853Sigor@sysoev.ru     nxt_queue_init(&tmcf->keeping);
74953Sigor@sysoev.ru     nxt_queue_init(&tmcf->updating);
75053Sigor@sysoev.ru     nxt_queue_init(&tmcf->pending);
75153Sigor@sysoev.ru     nxt_queue_init(&tmcf->creating);
752133Sigor@sysoev.ru     nxt_queue_init(&tmcf->apps);
753133Sigor@sysoev.ru     nxt_queue_init(&tmcf->previous);
75453Sigor@sysoev.ru 
75553Sigor@sysoev.ru     return tmcf;
75653Sigor@sysoev.ru 
75753Sigor@sysoev.ru temp_fail:
75853Sigor@sysoev.ru 
75965Sigor@sysoev.ru     nxt_mp_destroy(tmp);
76053Sigor@sysoev.ru 
76153Sigor@sysoev.ru fail:
76253Sigor@sysoev.ru 
76365Sigor@sysoev.ru     nxt_mp_destroy(mp);
76453Sigor@sysoev.ru 
76553Sigor@sysoev.ru     return NULL;
76653Sigor@sysoev.ru }
76753Sigor@sysoev.ru 
76853Sigor@sysoev.ru 
769198Sigor@sysoev.ru static void
770198Sigor@sysoev.ru nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data)
771139Sigor@sysoev.ru {
772139Sigor@sysoev.ru     nxt_int_t                    ret;
773139Sigor@sysoev.ru     nxt_router_t                 *router;
774139Sigor@sysoev.ru     nxt_runtime_t                *rt;
775198Sigor@sysoev.ru     nxt_queue_link_t             *qlk;
776198Sigor@sysoev.ru     nxt_socket_conf_t            *skcf;
777198Sigor@sysoev.ru     nxt_router_temp_conf_t       *tmcf;
778139Sigor@sysoev.ru     const nxt_event_interface_t  *interface;
779139Sigor@sysoev.ru 
780198Sigor@sysoev.ru     tmcf = obj;
781198Sigor@sysoev.ru 
782198Sigor@sysoev.ru     qlk = nxt_queue_first(&tmcf->pending);
783198Sigor@sysoev.ru 
784198Sigor@sysoev.ru     if (qlk != nxt_queue_tail(&tmcf->pending)) {
785198Sigor@sysoev.ru         nxt_queue_remove(qlk);
786198Sigor@sysoev.ru         nxt_queue_insert_tail(&tmcf->creating, qlk);
787198Sigor@sysoev.ru 
788198Sigor@sysoev.ru         skcf = nxt_queue_link_data(qlk, nxt_socket_conf_t, link);
789198Sigor@sysoev.ru 
790198Sigor@sysoev.ru         nxt_router_listen_socket_rpc_create(task, tmcf, skcf);
791198Sigor@sysoev.ru 
792198Sigor@sysoev.ru         return;
793139Sigor@sysoev.ru     }
794139Sigor@sysoev.ru 
795139Sigor@sysoev.ru     rt = task->thread->runtime;
796139Sigor@sysoev.ru 
797139Sigor@sysoev.ru     interface = nxt_service_get(rt->services, "engine", NULL);
798139Sigor@sysoev.ru 
799198Sigor@sysoev.ru     router = tmcf->conf->router;
800198Sigor@sysoev.ru 
801139Sigor@sysoev.ru     ret = nxt_router_engines_create(task, router, tmcf, interface);
802139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
803198Sigor@sysoev.ru         goto fail;
804139Sigor@sysoev.ru     }
805139Sigor@sysoev.ru 
806139Sigor@sysoev.ru     ret = nxt_router_threads_create(task, rt, tmcf);
807139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
808198Sigor@sysoev.ru         goto fail;
809139Sigor@sysoev.ru     }
810139Sigor@sysoev.ru 
811343Smax.romanov@nginx.com     nxt_router_apps_sort(task, router, tmcf);
812139Sigor@sysoev.ru 
813315Sigor@sysoev.ru     nxt_router_engines_post(router, tmcf);
814139Sigor@sysoev.ru 
815139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->updating);
816139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->creating);
817139Sigor@sysoev.ru 
818198Sigor@sysoev.ru     nxt_router_conf_ready(task, tmcf);
819198Sigor@sysoev.ru 
820198Sigor@sysoev.ru     return;
821198Sigor@sysoev.ru 
822198Sigor@sysoev.ru fail:
823