xref: /unit/src/nxt_router.c (revision 347)
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;
34346Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
35318Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
36318Smax.romanov@nginx.com 
37318Smax.romanov@nginx.com     nxt_queue_link_t     link;     /* for nxt_conn_t.requests */
38318Smax.romanov@nginx.com } nxt_req_conn_link_t;
39318Smax.romanov@nginx.com 
40318Smax.romanov@nginx.com 
41167Smax.romanov@nginx.com struct nxt_req_app_link_s {
42318Smax.romanov@nginx.com     uint32_t             stream;
43167Smax.romanov@nginx.com     nxt_port_t           *app_port;
44318Smax.romanov@nginx.com     nxt_pid_t            app_pid;
45167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
46167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
47167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
48167Smax.romanov@nginx.com 
49167Smax.romanov@nginx.com     nxt_queue_link_t     link; /* for nxt_app_t.requests */
50167Smax.romanov@nginx.com 
51167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
52167Smax.romanov@nginx.com     nxt_work_t           work;
53345Smax.romanov@nginx.com 
54345Smax.romanov@nginx.com     int                  err_code;
55345Smax.romanov@nginx.com     const char           *err_str;
56167Smax.romanov@nginx.com };
57167Smax.romanov@nginx.com 
58167Smax.romanov@nginx.com 
59198Sigor@sysoev.ru typedef struct {
60198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
61198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
62198Sigor@sysoev.ru } nxt_socket_rpc_t;
63198Sigor@sysoev.ru 
64198Sigor@sysoev.ru 
65343Smax.romanov@nginx.com static nxt_int_t nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app);
66343Smax.romanov@nginx.com 
67345Smax.romanov@nginx.com static void nxt_router_ra_error(nxt_task_t *task, nxt_req_app_link_t *ra,
68345Smax.romanov@nginx.com     int code, const char* str);
69345Smax.romanov@nginx.com 
70139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
71198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
72198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
73139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
74139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
75139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
76139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
77193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
7853Sigor@sysoev.ru static void nxt_router_listen_sockets_sort(nxt_router_t *router,
7953Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
8053Sigor@sysoev.ru 
81115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
82115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
83133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
84133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
85133Sigor@sysoev.ru     nxt_str_t *name);
86198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
87198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
88198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
89198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
90198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
91198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
9265Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task, nxt_mp_t *mp,
9365Sigor@sysoev.ru     nxt_sockaddr_t *sa);
9453Sigor@sysoev.ru 
9553Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
9653Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
9753Sigor@sysoev.ru     const nxt_event_interface_t *interface);
98115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
99115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
100115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
101115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
102115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
103115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
104115Sigor@sysoev.ru static void nxt_router_engine_socket_count(nxt_queue_t *sockets);
105154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
106154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
107154Sigor@sysoev.ru     nxt_work_handler_t handler);
108313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
109313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
110139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
111139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
11253Sigor@sysoev.ru 
11353Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
11453Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
11553Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
11653Sigor@sysoev.ru     nxt_event_engine_t *engine);
117343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
118133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
11953Sigor@sysoev.ru 
120315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
121315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
122315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
123315Sigor@sysoev.ru     nxt_work_t *jobs);
12453Sigor@sysoev.ru 
12553Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
12653Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
12753Sigor@sysoev.ru     void *data);
12853Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
12953Sigor@sysoev.ru     void *data);
13053Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
13153Sigor@sysoev.ru     void *data);
132313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
133313Sigor@sysoev.ru     void *data);
13453Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
13553Sigor@sysoev.ru     void *data);
13653Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
13753Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
13853Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
13953Sigor@sysoev.ru     void *data);
14053Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
14153Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
14253Sigor@sysoev.ru 
143343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
144343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
145343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
146343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
147343Smax.romanov@nginx.com 
148343Smax.romanov@nginx.com static nxt_port_t * nxt_router_app_get_idle_port(nxt_app_t *app);
149343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
150343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
151141Smax.romanov@nginx.com 
15253Sigor@sysoev.ru static void nxt_router_conn_init(nxt_task_t *task, void *obj, void *data);
15353Sigor@sysoev.ru static void nxt_router_conn_http_header_parse(nxt_task_t *task, void *obj,
15453Sigor@sysoev.ru     void *data);
155206Smax.romanov@nginx.com static void nxt_router_conn_http_body_read(nxt_task_t *task, void *obj,
156206Smax.romanov@nginx.com     void *data);
15788Smax.romanov@nginx.com static void nxt_router_process_http_request(nxt_task_t *task,
15888Smax.romanov@nginx.com     nxt_conn_t *c, nxt_app_parse_ctx_t *ap);
159141Smax.romanov@nginx.com static void nxt_router_process_http_request_mp(nxt_task_t *task,
160343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
161216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
162216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
163216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
164216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
165216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
166216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
16788Smax.romanov@nginx.com static void nxt_router_conn_ready(nxt_task_t *task, void *obj, void *data);
16853Sigor@sysoev.ru static void nxt_router_conn_close(nxt_task_t *task, void *obj, void *data);
16953Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
17053Sigor@sysoev.ru static void nxt_router_conn_error(nxt_task_t *task, void *obj, void *data);
17153Sigor@sysoev.ru static void nxt_router_conn_timeout(nxt_task_t *task, void *obj, void *data);
172318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
17362Sigor@sysoev.ru static nxt_msec_t nxt_router_conn_timeout_value(nxt_conn_t *c, uintptr_t data);
17420Sigor@sysoev.ru 
175141Smax.romanov@nginx.com static void nxt_router_gen_error(nxt_task_t *task, nxt_conn_t *c, int code,
176345Smax.romanov@nginx.com     const char* str);
177141Smax.romanov@nginx.com 
178119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
17920Sigor@sysoev.ru 
180216Sigor@sysoev.ru 
181216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
182216Sigor@sysoev.ru     nxt_python_prepare_msg,
183216Sigor@sysoev.ru     nxt_php_prepare_msg,
184216Sigor@sysoev.ru     nxt_go_prepare_msg,
185216Sigor@sysoev.ru };
186216Sigor@sysoev.ru 
187216Sigor@sysoev.ru 
18820Sigor@sysoev.ru nxt_int_t
189141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
19020Sigor@sysoev.ru {
191141Smax.romanov@nginx.com     nxt_int_t      ret;
192141Smax.romanov@nginx.com     nxt_router_t   *router;
193141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
194141Smax.romanov@nginx.com 
195141Smax.romanov@nginx.com     rt = task->thread->runtime;
19653Sigor@sysoev.ru 
19788Smax.romanov@nginx.com     ret = nxt_app_http_init(task, rt);
19888Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
19988Smax.romanov@nginx.com         return ret;
20088Smax.romanov@nginx.com     }
20188Smax.romanov@nginx.com 
20253Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
20353Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
20453Sigor@sysoev.ru         return NXT_ERROR;
20553Sigor@sysoev.ru     }
20653Sigor@sysoev.ru 
20753Sigor@sysoev.ru     nxt_queue_init(&router->engines);
20853Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
209133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
21053Sigor@sysoev.ru 
211119Smax.romanov@nginx.com     nxt_router = router;
212119Smax.romanov@nginx.com 
213115Sigor@sysoev.ru     return NXT_OK;
214115Sigor@sysoev.ru }
215115Sigor@sysoev.ru 
216115Sigor@sysoev.ru 
217343Smax.romanov@nginx.com static void
218343Smax.romanov@nginx.com nxt_router_start_worker_handler(nxt_task_t *task, nxt_port_t *port, void *data)
219167Smax.romanov@nginx.com {
220343Smax.romanov@nginx.com     size_t         size;
221343Smax.romanov@nginx.com     uint32_t       stream;
222343Smax.romanov@nginx.com     nxt_app_t      *app;
223343Smax.romanov@nginx.com     nxt_buf_t      *b;
224343Smax.romanov@nginx.com     nxt_port_t     *main_port;
225343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
226343Smax.romanov@nginx.com 
227343Smax.romanov@nginx.com     app = data;
228167Smax.romanov@nginx.com 
229167Smax.romanov@nginx.com     rt = task->thread->runtime;
230240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
231167Smax.romanov@nginx.com 
232343Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start worker", &app->name, app);
233343Smax.romanov@nginx.com 
234343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
235343Smax.romanov@nginx.com 
236343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
237343Smax.romanov@nginx.com 
238343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
239343Smax.romanov@nginx.com         goto failed;
240167Smax.romanov@nginx.com     }
241167Smax.romanov@nginx.com 
242343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
243343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
244343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
245343Smax.romanov@nginx.com 
246343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
247343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
248343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
249343Smax.romanov@nginx.com                                            -1, app);
250343Smax.romanov@nginx.com 
251343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
252343Smax.romanov@nginx.com         nxt_mp_release(b->data, b);
253343Smax.romanov@nginx.com 
254343Smax.romanov@nginx.com         goto failed;
255343Smax.romanov@nginx.com     }
256343Smax.romanov@nginx.com 
257343Smax.romanov@nginx.com     nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
258343Smax.romanov@nginx.com                           stream, port->id, b);
259343Smax.romanov@nginx.com 
260343Smax.romanov@nginx.com     return;
261343Smax.romanov@nginx.com 
262343Smax.romanov@nginx.com failed:
263343Smax.romanov@nginx.com 
264343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
265343Smax.romanov@nginx.com 
266343Smax.romanov@nginx.com     app->pending_workers--;
267343Smax.romanov@nginx.com 
268343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
269343Smax.romanov@nginx.com 
270343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
271167Smax.romanov@nginx.com }
272167Smax.romanov@nginx.com 
273167Smax.romanov@nginx.com 
274343Smax.romanov@nginx.com static nxt_int_t
275343Smax.romanov@nginx.com nxt_router_start_worker(nxt_task_t *task, nxt_app_t *app)
276141Smax.romanov@nginx.com {
277343Smax.romanov@nginx.com     nxt_int_t      res;
278343Smax.romanov@nginx.com     nxt_port_t     *router_port;
279343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
280343Smax.romanov@nginx.com 
281343Smax.romanov@nginx.com     rt = task->thread->runtime;
282343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
283343Smax.romanov@nginx.com 
284343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
285343Smax.romanov@nginx.com 
286343Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_worker_handler,
287343Smax.romanov@nginx.com                         app);
288343Smax.romanov@nginx.com 
289343Smax.romanov@nginx.com     if (res == NXT_OK) {
290343Smax.romanov@nginx.com         return res;
291318Smax.romanov@nginx.com     }
292318Smax.romanov@nginx.com 
293343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
294343Smax.romanov@nginx.com 
295343Smax.romanov@nginx.com     app->pending_workers--;
296343Smax.romanov@nginx.com 
297343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
298343Smax.romanov@nginx.com 
299343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
300343Smax.romanov@nginx.com 
301343Smax.romanov@nginx.com     return NXT_ERROR;
302318Smax.romanov@nginx.com }
303318Smax.romanov@nginx.com 
304318Smax.romanov@nginx.com 
305167Smax.romanov@nginx.com static nxt_req_app_link_t *
306167Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_conn_link_t *rc)
307167Smax.romanov@nginx.com {
308167Smax.romanov@nginx.com     nxt_mp_t            *mp;
309318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
310167Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
311167Smax.romanov@nginx.com 
312346Smax.romanov@nginx.com     mp = rc->ap->mem_pool;
313318Smax.romanov@nginx.com     engine = task->thread->engine;
314167Smax.romanov@nginx.com 
315167Smax.romanov@nginx.com     ra = nxt_mp_retain(mp, sizeof(nxt_req_app_link_t));
316167Smax.romanov@nginx.com 
317167Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
318167Smax.romanov@nginx.com         return NULL;
319167Smax.romanov@nginx.com     }
320167Smax.romanov@nginx.com 
321318Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD create", rc->stream);
322167Smax.romanov@nginx.com 
323167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
324167Smax.romanov@nginx.com 
325318Smax.romanov@nginx.com     ra->stream = rc->stream;
326318Smax.romanov@nginx.com     ra->app_pid = -1;
327167Smax.romanov@nginx.com     ra->rc = rc;
328318Smax.romanov@nginx.com     rc->ra = ra;
329318Smax.romanov@nginx.com     ra->reply_port = engine->port;
330167Smax.romanov@nginx.com 
331167Smax.romanov@nginx.com     ra->mem_pool = mp;
332167Smax.romanov@nginx.com 
333167Smax.romanov@nginx.com     ra->work.handler = NULL;
334318Smax.romanov@nginx.com     ra->work.task = &engine->task;
335167Smax.romanov@nginx.com     ra->work.obj = ra;
336318Smax.romanov@nginx.com     ra->work.data = engine;
337167Smax.romanov@nginx.com 
338167Smax.romanov@nginx.com     return ra;
339167Smax.romanov@nginx.com }
340167Smax.romanov@nginx.com 
341167Smax.romanov@nginx.com 
342167Smax.romanov@nginx.com static void
343167Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, void *obj, void *data)
344167Smax.romanov@nginx.com {
345343Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
346343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
347343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
348318Smax.romanov@nginx.com 
349318Smax.romanov@nginx.com     ra = obj;
350318Smax.romanov@nginx.com     engine = data;
351318Smax.romanov@nginx.com 
352343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
353343Smax.romanov@nginx.com         if (ra->app_port != NULL) {
354343Smax.romanov@nginx.com             ra->app_pid = ra->app_port->pid;
355318Smax.romanov@nginx.com         }
356318Smax.romanov@nginx.com 
357318Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release;
358318Smax.romanov@nginx.com         ra->work.task = &engine->task;
359318Smax.romanov@nginx.com         ra->work.next = NULL;
360318Smax.romanov@nginx.com 
361318Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
362318Smax.romanov@nginx.com                   ra->stream, engine);
363318Smax.romanov@nginx.com 
364318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
365318Smax.romanov@nginx.com 
366318Smax.romanov@nginx.com         return;
367318Smax.romanov@nginx.com     }
368318Smax.romanov@nginx.com 
369343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
370343Smax.romanov@nginx.com 
371343Smax.romanov@nginx.com     rc = ra->rc;
372343Smax.romanov@nginx.com 
373343Smax.romanov@nginx.com     if (rc != NULL) {
374343Smax.romanov@nginx.com         if (ra->app_pid != -1) {
375343Smax.romanov@nginx.com             nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_pid);
376343Smax.romanov@nginx.com         }
377343Smax.romanov@nginx.com 
378343Smax.romanov@nginx.com         rc->app_port = ra->app_port;
379343Smax.romanov@nginx.com 
380343Smax.romanov@nginx.com         ra->app_port = NULL;
381343Smax.romanov@nginx.com         rc->ra = NULL;
382343Smax.romanov@nginx.com         ra->rc = NULL;
383318Smax.romanov@nginx.com     }
384318Smax.romanov@nginx.com 
385343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
386343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
387343Smax.romanov@nginx.com 
388343Smax.romanov@nginx.com         ra->app_port = NULL;
389343Smax.romanov@nginx.com     }
390318Smax.romanov@nginx.com 
391318Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
392318Smax.romanov@nginx.com }
393318Smax.romanov@nginx.com 
394318Smax.romanov@nginx.com 
395318Smax.romanov@nginx.com static void
396318Smax.romanov@nginx.com nxt_router_ra_abort(nxt_task_t *task, void *obj, void *data)
397318Smax.romanov@nginx.com {
398343Smax.romanov@nginx.com     nxt_conn_t           *c;
399343Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
400343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
401343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
402167Smax.romanov@nginx.com 
403167Smax.romanov@nginx.com     ra = obj;
404167Smax.romanov@nginx.com     engine = data;
405167Smax.romanov@nginx.com 
406167Smax.romanov@nginx.com     if (task->thread->engine != engine) {
407318Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_abort;
408167Smax.romanov@nginx.com         ra->work.task = &engine->task;
409167Smax.romanov@nginx.com         ra->work.next = NULL;
410167Smax.romanov@nginx.com 
411318Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post abort to %p", ra->stream, engine);
412167Smax.romanov@nginx.com 
413167Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
414167Smax.romanov@nginx.com 
415167Smax.romanov@nginx.com         return;
416167Smax.romanov@nginx.com     }
417167Smax.romanov@nginx.com 
418318Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD abort", ra->stream);
419318Smax.romanov@nginx.com 
420343Smax.romanov@nginx.com     rc = ra->rc;
421343Smax.romanov@nginx.com 
422343Smax.romanov@nginx.com     if (rc != NULL) {
423343Smax.romanov@nginx.com         c = rc->conn;
424318Smax.romanov@nginx.com 
425318Smax.romanov@nginx.com         nxt_router_gen_error(task, c, 500,
426318Smax.romanov@nginx.com                              "Failed to start application worker");
427343Smax.romanov@nginx.com 
428343Smax.romanov@nginx.com         rc->ra = NULL;
429343Smax.romanov@nginx.com         ra->rc = NULL;
430343Smax.romanov@nginx.com     }
431343Smax.romanov@nginx.com 
432343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
433343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
434343Smax.romanov@nginx.com 
435343Smax.romanov@nginx.com         ra->app_port = NULL;
436167Smax.romanov@nginx.com     }
437167Smax.romanov@nginx.com 
438167Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
439167Smax.romanov@nginx.com }
440167Smax.romanov@nginx.com 
441167Smax.romanov@nginx.com 
442345Smax.romanov@nginx.com static void
443345Smax.romanov@nginx.com nxt_router_ra_error_handler(nxt_task_t *task, void *obj, void *data)
444345Smax.romanov@nginx.com {
445345Smax.romanov@nginx.com     nxt_req_app_link_t   *ra;
446345Smax.romanov@nginx.com 
447345Smax.romanov@nginx.com     ra = obj;
448345Smax.romanov@nginx.com 
449345Smax.romanov@nginx.com     nxt_router_ra_error(task, ra, ra->err_code, ra->err_str);
450345Smax.romanov@nginx.com }
451345Smax.romanov@nginx.com 
452345Smax.romanov@nginx.com 
453345Smax.romanov@nginx.com static void
454345Smax.romanov@nginx.com nxt_router_ra_error(nxt_task_t *task, nxt_req_app_link_t *ra, int code,
455345Smax.romanov@nginx.com     const char* str)
456345Smax.romanov@nginx.com {
457345Smax.romanov@nginx.com     nxt_conn_t           *c;
458345Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
459345Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
460345Smax.romanov@nginx.com 
461345Smax.romanov@nginx.com     engine = ra->work.data;
462345Smax.romanov@nginx.com 
463345Smax.romanov@nginx.com     if (task->thread->engine != engine) {
464345Smax.romanov@nginx.com         ra->err_code = code;
465345Smax.romanov@nginx.com         ra->err_str = str;
466345Smax.romanov@nginx.com 
467345Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_error_handler;
468345Smax.romanov@nginx.com         ra->work.task = &engine->task;
469345Smax.romanov@nginx.com         ra->work.next = NULL;
470345Smax.romanov@nginx.com 
471345Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post error to %p", ra->stream, engine);
472345Smax.romanov@nginx.com 
473345Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
474345Smax.romanov@nginx.com 
475345Smax.romanov@nginx.com         return;
476345Smax.romanov@nginx.com     }
477345Smax.romanov@nginx.com 
478345Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD error", ra->stream);
479345Smax.romanov@nginx.com 
480345Smax.romanov@nginx.com     rc = ra->rc;
481345Smax.romanov@nginx.com 
482345Smax.romanov@nginx.com     if (rc != NULL) {
483345Smax.romanov@nginx.com         c = rc->conn;
484345Smax.romanov@nginx.com 
485345Smax.romanov@nginx.com         nxt_router_gen_error(task, c, code, str);
486345Smax.romanov@nginx.com 
487345Smax.romanov@nginx.com         rc->ra = NULL;
488345Smax.romanov@nginx.com         ra->rc = NULL;
489345Smax.romanov@nginx.com     }
490345Smax.romanov@nginx.com 
491345Smax.romanov@nginx.com     if (ra->app_port != NULL) {
492345Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
493345Smax.romanov@nginx.com 
494345Smax.romanov@nginx.com         ra->app_port = NULL;
495345Smax.romanov@nginx.com     }
496345Smax.romanov@nginx.com 
497345Smax.romanov@nginx.com     nxt_mp_release(ra->mem_pool, ra);
498345Smax.romanov@nginx.com }
499345Smax.romanov@nginx.com 
500345Smax.romanov@nginx.com 
501343Smax.romanov@nginx.com nxt_inline void
502343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
503343Smax.romanov@nginx.com {
504343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
505343Smax.romanov@nginx.com 
506343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
507343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
508343Smax.romanov@nginx.com 
509343Smax.romanov@nginx.com         rc->app_port = NULL;
510343Smax.romanov@nginx.com     }
511343Smax.romanov@nginx.com 
512343Smax.romanov@nginx.com     ra = rc->ra;
513343Smax.romanov@nginx.com 
514343Smax.romanov@nginx.com     if (ra != NULL) {
515343Smax.romanov@nginx.com         rc->ra = NULL;
516343Smax.romanov@nginx.com         ra->rc = NULL;
517343Smax.romanov@nginx.com 
518343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
519343Smax.romanov@nginx.com 
520343Smax.romanov@nginx.com         if (ra->link.next != NULL) {
521343Smax.romanov@nginx.com             nxt_queue_remove(&ra->link);
522343Smax.romanov@nginx.com 
523343Smax.romanov@nginx.com             ra->link.next = NULL;
524343Smax.romanov@nginx.com 
525343Smax.romanov@nginx.com         } else {
526343Smax.romanov@nginx.com             ra = NULL;
527343Smax.romanov@nginx.com         }
528343Smax.romanov@nginx.com 
529343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
530343Smax.romanov@nginx.com     }
531343Smax.romanov@nginx.com 
532343Smax.romanov@nginx.com     if (ra != NULL) {
533343Smax.romanov@nginx.com         nxt_router_ra_release(task, ra, ra->work.data);
534343Smax.romanov@nginx.com     }
535343Smax.romanov@nginx.com 
536343Smax.romanov@nginx.com     if (rc->app != NULL) {
537343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
538343Smax.romanov@nginx.com 
539343Smax.romanov@nginx.com         rc->app = NULL;
540343Smax.romanov@nginx.com     }
541343Smax.romanov@nginx.com 
542346Smax.romanov@nginx.com     if (rc->ap != NULL) {
543346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
544346Smax.romanov@nginx.com 
545346Smax.romanov@nginx.com         rc->ap = NULL;
546346Smax.romanov@nginx.com     }
547346Smax.romanov@nginx.com 
548343Smax.romanov@nginx.com     nxt_queue_remove(&rc->link);
549343Smax.romanov@nginx.com 
550343Smax.romanov@nginx.com     rc->conn = NULL;
551343Smax.romanov@nginx.com }
552343Smax.romanov@nginx.com 
553343Smax.romanov@nginx.com 
554141Smax.romanov@nginx.com void
555141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
556141Smax.romanov@nginx.com {
557141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
558141Smax.romanov@nginx.com 
559192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
560141Smax.romanov@nginx.com         return;
561141Smax.romanov@nginx.com     }
562141Smax.romanov@nginx.com 
563*347Smax.romanov@nginx.com     if (msg->u.new_port == NULL ||
564*347Smax.romanov@nginx.com         msg->u.new_port->type != NXT_PROCESS_WORKER)
565*347Smax.romanov@nginx.com     {
566192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
567141Smax.romanov@nginx.com     }
568192Smax.romanov@nginx.com 
569192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
570141Smax.romanov@nginx.com }
571141Smax.romanov@nginx.com 
572141Smax.romanov@nginx.com 
573139Sigor@sysoev.ru void
574139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
575115Sigor@sysoev.ru {
576139Sigor@sysoev.ru     size_t                  dump_size;
577198Sigor@sysoev.ru     nxt_int_t               ret;
578139Sigor@sysoev.ru     nxt_buf_t               *b;
579139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
580139Sigor@sysoev.ru 
581139Sigor@sysoev.ru     b = msg->buf;
582139Sigor@sysoev.ru 
583139Sigor@sysoev.ru     dump_size = nxt_buf_used_size(b);
584139Sigor@sysoev.ru 
585139Sigor@sysoev.ru     if (dump_size > 300) {
586139Sigor@sysoev.ru         dump_size = 300;
58753Sigor@sysoev.ru     }
58853Sigor@sysoev.ru 
589139Sigor@sysoev.ru     nxt_debug(task, "router conf data (%z): %*s",
590139Sigor@sysoev.ru               msg->size, dump_size, b->mem.pos);
591139Sigor@sysoev.ru 
592139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
593139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
594139Sigor@sysoev.ru         return;
59553Sigor@sysoev.ru     }
59653Sigor@sysoev.ru 
597139Sigor@sysoev.ru     tmcf->conf->router = nxt_router;
598139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
599139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
600198Sigor@sysoev.ru                                        msg->port_msg.pid,
601198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
602198Sigor@sysoev.ru 
603198Sigor@sysoev.ru     ret = nxt_router_conf_create(task, tmcf, b->mem.pos, b->mem.free);
604198Sigor@sysoev.ru 
605198Sigor@sysoev.ru     if (nxt_fast_path(ret == NXT_OK)) {
606198Sigor@sysoev.ru         nxt_router_conf_apply(task, tmcf, NULL);
607198Sigor@sysoev.ru 
608198Sigor@sysoev.ru     } else {
609198Sigor@sysoev.ru         nxt_router_conf_error(task, tmcf);
610139Sigor@sysoev.ru     }
61153Sigor@sysoev.ru }
61253Sigor@sysoev.ru 
61353Sigor@sysoev.ru 
614*347Smax.romanov@nginx.com static void
615*347Smax.romanov@nginx.com nxt_router_worker_remove_pid(nxt_task_t *task, nxt_port_t *port, void *data)
616*347Smax.romanov@nginx.com {
617*347Smax.romanov@nginx.com     union {
618*347Smax.romanov@nginx.com         nxt_pid_t  removed_pid;
619*347Smax.romanov@nginx.com         void       *data;
620*347Smax.romanov@nginx.com     } u;
621*347Smax.romanov@nginx.com 
622*347Smax.romanov@nginx.com     u.data = data;
623*347Smax.romanov@nginx.com 
624*347Smax.romanov@nginx.com     nxt_port_rpc_remove_peer(task, port, u.removed_pid);
625*347Smax.romanov@nginx.com }
626*347Smax.romanov@nginx.com 
627*347Smax.romanov@nginx.com 
628192Smax.romanov@nginx.com void
629192Smax.romanov@nginx.com nxt_router_remove_pid_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
630192Smax.romanov@nginx.com {
631*347Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
632318Smax.romanov@nginx.com 
633192Smax.romanov@nginx.com     nxt_port_remove_pid_handler(task, msg);
634192Smax.romanov@nginx.com 
635192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
636192Smax.romanov@nginx.com         return;
637192Smax.romanov@nginx.com     }
638192Smax.romanov@nginx.com 
639318Smax.romanov@nginx.com     nxt_queue_each(engine, &nxt_router->engines, nxt_event_engine_t, link0)
640318Smax.romanov@nginx.com     {
641*347Smax.romanov@nginx.com         nxt_port_post(task, engine->port, nxt_router_worker_remove_pid,
642*347Smax.romanov@nginx.com                       msg->u.data);
643318Smax.romanov@nginx.com     }
644318Smax.romanov@nginx.com     nxt_queue_loop;
645318Smax.romanov@nginx.com 
646192Smax.romanov@nginx.com     msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
647192Smax.romanov@nginx.com 
648192Smax.romanov@nginx.com     nxt_port_rpc_handler(task, msg);
649192Smax.romanov@nginx.com }
650192Smax.romanov@nginx.com 
651192Smax.romanov@nginx.com 
65253Sigor@sysoev.ru static nxt_router_temp_conf_t *
653139Sigor@sysoev.ru nxt_router_temp_conf(nxt_task_t *task)
65453Sigor@sysoev.ru {
65565Sigor@sysoev.ru     nxt_mp_t                *mp, *tmp;
65653Sigor@sysoev.ru     nxt_router_conf_t       *rtcf;
65753Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
65853Sigor@sysoev.ru 
65965Sigor@sysoev.ru     mp = nxt_mp_create(1024, 128, 256, 32);
66053Sigor@sysoev.ru     if (nxt_slow_path(mp == NULL)) {
66153Sigor@sysoev.ru         return NULL;
66253Sigor@sysoev.ru     }
66353Sigor@sysoev.ru 
66465Sigor@sysoev.ru     rtcf = nxt_mp_zget(mp, sizeof(nxt_router_conf_t));
66553Sigor@sysoev.ru     if (nxt_slow_path(rtcf == NULL)) {
66653Sigor@sysoev.ru         goto fail;
66753Sigor@sysoev.ru     }
66853Sigor@sysoev.ru 
66953Sigor@sysoev.ru     rtcf->mem_pool = mp;
67053Sigor@sysoev.ru 
67165Sigor@sysoev.ru     tmp = nxt_mp_create(1024, 128, 256, 32);
67253Sigor@sysoev.ru     if (nxt_slow_path(tmp == NULL)) {
67353Sigor@sysoev.ru         goto fail;
67453Sigor@sysoev.ru     }
67553Sigor@sysoev.ru 
67665Sigor@sysoev.ru     tmcf = nxt_mp_zget(tmp, sizeof(nxt_router_temp_conf_t));
67753Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
67853Sigor@sysoev.ru         goto temp_fail;
67953Sigor@sysoev.ru     }
68053Sigor@sysoev.ru 
68153Sigor@sysoev.ru     tmcf->mem_pool = tmp;
68253Sigor@sysoev.ru     tmcf->conf = rtcf;
683139Sigor@sysoev.ru     tmcf->count = 1;
684139Sigor@sysoev.ru     tmcf->engine = task->thread->engine;
68553Sigor@sysoev.ru 
68653Sigor@sysoev.ru     tmcf->engines = nxt_array_create(tmcf->mem_pool, 4,
68753Sigor@sysoev.ru                                      sizeof(nxt_router_engine_conf_t));
68853Sigor@sysoev.ru     if (nxt_slow_path(tmcf->engines == NULL)) {
68953Sigor@sysoev.ru         goto temp_fail;
69053Sigor@sysoev.ru     }
69153Sigor@sysoev.ru 
69253Sigor@sysoev.ru     nxt_queue_init(&tmcf->deleting);
69353Sigor@sysoev.ru     nxt_queue_init(&tmcf->keeping);
69453Sigor@sysoev.ru     nxt_queue_init(&tmcf->updating);
69553Sigor@sysoev.ru     nxt_queue_init(&tmcf->pending);
69653Sigor@sysoev.ru     nxt_queue_init(&tmcf->creating);
697133Sigor@sysoev.ru     nxt_queue_init(&tmcf->apps);
698133Sigor@sysoev.ru     nxt_queue_init(&tmcf->previous);
69953Sigor@sysoev.ru 
70053Sigor@sysoev.ru     return tmcf;
70153Sigor@sysoev.ru 
70253Sigor@sysoev.ru temp_fail:
70353Sigor@sysoev.ru 
70465Sigor@sysoev.ru     nxt_mp_destroy(tmp);
70553Sigor@sysoev.ru 
70653Sigor@sysoev.ru fail:
70753Sigor@sysoev.ru 
70865Sigor@sysoev.ru     nxt_mp_destroy(mp);
70953Sigor@sysoev.ru 
71053Sigor@sysoev.ru     return NULL;
71153Sigor@sysoev.ru }
71253Sigor@sysoev.ru 
71353Sigor@sysoev.ru 
714198Sigor@sysoev.ru static void
715198Sigor@sysoev.ru nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data)
716139Sigor@sysoev.ru {
717139Sigor@sysoev.ru     nxt_int_t                    ret;
718139Sigor@sysoev.ru     nxt_router_t                 *router;
719139Sigor@sysoev.ru     nxt_runtime_t                *rt;
720198Sigor@sysoev.ru     nxt_queue_link_t             *qlk;
721198Sigor@sysoev.ru     nxt_socket_conf_t            *skcf;
722198Sigor@sysoev.ru     nxt_router_temp_conf_t       *tmcf;
723139Sigor@sysoev.ru     const nxt_event_interface_t  *interface;
724139Sigor@sysoev.ru 
725198Sigor@sysoev.ru     tmcf = obj;
726198Sigor@sysoev.ru 
727198Sigor@sysoev.ru     qlk = nxt_queue_first(&tmcf->pending);
728198Sigor@sysoev.ru 
729198Sigor@sysoev.ru     if (qlk != nxt_queue_tail(&tmcf->pending)) {
730198Sigor@sysoev.ru         nxt_queue_remove(qlk);
731198Sigor@sysoev.ru         nxt_queue_insert_tail(&tmcf->creating, qlk);
732198Sigor@sysoev.ru 
733198Sigor@sysoev.ru         skcf = nxt_queue_link_data(qlk, nxt_socket_conf_t, link);
734198Sigor@sysoev.ru 
735198Sigor@sysoev.ru         nxt_router_listen_socket_rpc_create(task, tmcf, skcf);
736198Sigor@sysoev.ru 
737198Sigor@sysoev.ru         return;
738139Sigor@sysoev.ru     }
739139Sigor@sysoev.ru 
740139Sigor@sysoev.ru     rt = task->thread->runtime;
741139Sigor@sysoev.ru 
742139Sigor@sysoev.ru     interface = nxt_service_get(rt->services, "engine", NULL);
743139Sigor@sysoev.ru 
744198Sigor@sysoev.ru     router = tmcf->conf->router;
745198Sigor@sysoev.ru 
746139Sigor@sysoev.ru     ret = nxt_router_engines_create(task, router, tmcf, interface);
747139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
748198Sigor@sysoev.ru         goto fail;
749139Sigor@sysoev.ru     }
750139Sigor@sysoev.ru 
751139Sigor@sysoev.ru     ret = nxt_router_threads_create(task, rt, tmcf);
752139Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
753198Sigor@sysoev.ru         goto fail;
754139Sigor@sysoev.ru     }
755139Sigor@sysoev.ru 
756343Smax.romanov@nginx.com     nxt_router_apps_sort(task, router, tmcf);
757139Sigor@sysoev.ru 
758315Sigor@sysoev.ru     nxt_router_engines_post(router, tmcf);
759139Sigor@sysoev.ru 
760139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->updating);
761139Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->creating);
762139Sigor@sysoev.ru 
763198Sigor@sysoev.ru     nxt_router_conf_ready(task, tmcf);
764198Sigor@sysoev.ru 
765198Sigor@sysoev.ru     return;
766198Sigor@sysoev.ru 
767198Sigor@sysoev.ru fail:
768198Sigor@sysoev.ru 
769198Sigor@sysoev.ru     nxt_router_conf_error(task, tmcf);
770198Sigor@sysoev.ru 
771198Sigor@sysoev.ru     return;
772139Sigor@sysoev.ru }
773139Sigor@sysoev.ru 
774139Sigor@sysoev.ru 
775139Sigor@sysoev.ru static void
776139Sigor@sysoev.ru nxt_router_conf_wait(nxt_task_t *task, void *obj, void *data)
777139Sigor@sysoev.ru {
778153Sigor@sysoev.ru     nxt_joint_job_t  *job;
779153Sigor@sysoev.ru 
780153Sigor@sysoev.ru     job = obj;
781153Sigor@sysoev.ru 
782198Sigor@sysoev.ru     nxt_router_conf_ready(task, job->tmcf);
783139Sigor@sysoev.ru }
784139Sigor@sysoev.ru 
785139Sigor@sysoev.ru 
786139Sigor@sysoev.ru static void
787198Sigor@sysoev.ru nxt_router_conf_ready(nxt_task_t *task, nxt_router_temp_conf_t *tmcf)
788139Sigor@sysoev.ru {
789139Sigor@sysoev.ru     nxt_debug(task, "temp conf count:%D", tmcf->count);
790139Sigor@sysoev.ru 
791139Sigor@sysoev.ru     if (--tmcf->count == 0) {
792193Smax.romanov@nginx.com         nxt_router_conf_send(task, tmcf, NXT_PORT_MSG_RPC_READY_LAST);
793139Sigor@sysoev.ru     }
794139Sigor@sysoev.ru }
795139Sigor@sysoev.ru 
796139Sigor@sysoev.ru 
797139Sigor@sysoev.ru static void
798139Sigor@sysoev.ru nxt_router_conf_error(nxt_task_t *task, nxt_router_temp_conf_t *tmcf)
799139Sigor@sysoev.ru {
800148Sigor@sysoev.ru     nxt_socket_t       s;
801149Sigor@sysoev.ru     nxt_router_t       *router;
802148Sigor@sysoev.ru     nxt_queue_link_t   *qlk;
803148Sigor@sysoev.ru     nxt_socket_conf_t  *skcf;
804148Sigor@sysoev.ru 
805198Sigor@sysoev.ru     nxt_log(task, NXT_LOG_CRIT, "failed to apply new conf");
806198Sigor@sysoev.ru 
807148Sigor@sysoev.ru     for (qlk = nxt_queue_first(&tmcf->creating);
808148Sigor@sysoev.ru          qlk != nxt_queue_tail(&tmcf->creating);
809148Sigor@sysoev.ru          qlk = nxt_queue_next(qlk))
810148Sigor@sysoev.ru     {
811148Sigor@sysoev.ru         skcf = nxt_queue_link_data(qlk, nxt_socket_conf_t, link);
812148Sigor@sysoev.ru         s = skcf->listen.socket;
813148Sigor@sysoev.ru 
814148Sigor@sysoev.ru         if (s != -1) {
815148Sigor@sysoev.ru             nxt_socket_close(task, s);
816148Sigor@sysoev.ru         }
817148Sigor@sysoev.ru 
818148Sigor@sysoev.ru         nxt_free(skcf->socket);
819148Sigor@sysoev.ru     }
820148Sigor@sysoev.ru 
821149Sigor@sysoev.ru     router = tmcf->conf->router;
822149Sigor@sysoev.ru 
823149Sigor@sysoev.ru     nxt_queue_add(&router->sockets, &tmcf->keeping);
824