xref: /unit/src/nxt_router.c (revision 1561)
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>
10774Svbart@nginx.com #if (NXT_TLS)
11774Svbart@nginx.com #include <nxt_cert.h>
12774Svbart@nginx.com #endif
13431Sigor@sysoev.ru #include <nxt_http.h>
14743Smax.romanov@nginx.com #include <nxt_port_memory_int.h>
15743Smax.romanov@nginx.com #include <nxt_unit_request.h>
16743Smax.romanov@nginx.com #include <nxt_unit_response.h>
171131Smax.romanov@nginx.com #include <nxt_router_request.h>
181555Smax.romanov@nginx.com #include <nxt_app_queue.h>
191555Smax.romanov@nginx.com #include <nxt_port_queue.h>
2020Sigor@sysoev.ru 
21115Sigor@sysoev.ru typedef struct {
22318Smax.romanov@nginx.com     nxt_str_t         type;
23507Smax.romanov@nginx.com     uint32_t          processes;
24507Smax.romanov@nginx.com     uint32_t          max_processes;
25507Smax.romanov@nginx.com     uint32_t          spare_processes;
26318Smax.romanov@nginx.com     nxt_msec_t        timeout;
27427Smax.romanov@nginx.com     nxt_msec_t        res_timeout;
28507Smax.romanov@nginx.com     nxt_msec_t        idle_timeout;
29318Smax.romanov@nginx.com     uint32_t          requests;
30318Smax.romanov@nginx.com     nxt_conf_value_t  *limits_value;
31507Smax.romanov@nginx.com     nxt_conf_value_t  *processes_value;
321473Svbart@nginx.com     nxt_conf_value_t  *targets_value;
33133Sigor@sysoev.ru } nxt_router_app_conf_t;
34133Sigor@sysoev.ru 
35133Sigor@sysoev.ru 
36133Sigor@sysoev.ru typedef struct {
37964Sigor@sysoev.ru     nxt_str_t         pass;
38964Sigor@sysoev.ru     nxt_str_t         application;
39115Sigor@sysoev.ru } nxt_router_listener_conf_t;
40115Sigor@sysoev.ru 
41115Sigor@sysoev.ru 
42774Svbart@nginx.com #if (NXT_TLS)
43774Svbart@nginx.com 
44774Svbart@nginx.com typedef struct {
45774Svbart@nginx.com     nxt_str_t          name;
46774Svbart@nginx.com     nxt_socket_conf_t  *conf;
47774Svbart@nginx.com 
48774Svbart@nginx.com     nxt_queue_link_t   link;  /* for nxt_socket_conf_t.tls */
49774Svbart@nginx.com } nxt_router_tlssock_t;
50774Svbart@nginx.com 
51774Svbart@nginx.com #endif
52774Svbart@nginx.com 
53774Svbart@nginx.com 
54198Sigor@sysoev.ru typedef struct {
55198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
56198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
57198Sigor@sysoev.ru } nxt_socket_rpc_t;
58198Sigor@sysoev.ru 
59198Sigor@sysoev.ru 
60507Smax.romanov@nginx.com typedef struct {
61507Smax.romanov@nginx.com     nxt_app_t               *app;
62507Smax.romanov@nginx.com     nxt_router_temp_conf_t  *temp_conf;
63507Smax.romanov@nginx.com } nxt_app_rpc_t;
64507Smax.romanov@nginx.com 
65507Smax.romanov@nginx.com 
661488St.nateldemoura@f5.com static nxt_int_t nxt_router_prefork(nxt_task_t *task, nxt_process_t *process,
671488St.nateldemoura@f5.com     nxt_mp_t *mp);
681488St.nateldemoura@f5.com static nxt_int_t nxt_router_start(nxt_task_t *task, nxt_process_data_t *data);
69662Smax.romanov@nginx.com static void nxt_router_greet_controller(nxt_task_t *task,
70662Smax.romanov@nginx.com     nxt_port_t *controller_port);
71662Smax.romanov@nginx.com 
72507Smax.romanov@nginx.com static nxt_int_t nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app);
73425Smax.romanov@nginx.com 
741552Smax.romanov@nginx.com static void nxt_router_new_port_handler(nxt_task_t *task,
751552Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
761552Smax.romanov@nginx.com static void nxt_router_conf_data_handler(nxt_task_t *task,
771552Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
781552Smax.romanov@nginx.com static void nxt_router_remove_pid_handler(nxt_task_t *task,
791552Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
801552Smax.romanov@nginx.com static void nxt_router_access_log_reopen_handler(nxt_task_t *task,
811552Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
821552Smax.romanov@nginx.com 
83139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
84198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
85198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
86139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
87139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
88139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
89139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
90193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
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);
941183Svbart@nginx.com static nxt_int_t nxt_router_conf_process_static(nxt_task_t *task,
951183Svbart@nginx.com     nxt_router_conf_t *rtcf, nxt_conf_value_t *conf);
96133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
971555Smax.romanov@nginx.com static nxt_int_t nxt_router_app_queue_init(nxt_task_t *task,
981555Smax.romanov@nginx.com     nxt_port_t *port);
991555Smax.romanov@nginx.com static nxt_int_t nxt_router_port_queue_init(nxt_task_t *task,
1001555Smax.romanov@nginx.com     nxt_port_t *port);
1011555Smax.romanov@nginx.com static nxt_int_t nxt_router_port_queue_map(nxt_task_t *task,
1021555Smax.romanov@nginx.com     nxt_port_t *port, nxt_fd_t fd);
103198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
104198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
105198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
106198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
107198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
108198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
109774Svbart@nginx.com #if (NXT_TLS)
110774Svbart@nginx.com static void nxt_router_tls_rpc_create(nxt_task_t *task,
111774Svbart@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_router_tlssock_t *tls);
112774Svbart@nginx.com static void nxt_router_tls_rpc_handler(nxt_task_t *task,
113774Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
114774Svbart@nginx.com #endif
115507Smax.romanov@nginx.com static void nxt_router_app_rpc_create(nxt_task_t *task,
116507Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_app_t *app);
117507Smax.romanov@nginx.com static void nxt_router_app_prefork_ready(nxt_task_t *task,
118507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
119507Smax.romanov@nginx.com static void nxt_router_app_prefork_error(nxt_task_t *task,
120507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
121359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
122359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
123359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
124359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
12553Sigor@sysoev.ru 
12653Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
12753Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
12853Sigor@sysoev.ru     const nxt_event_interface_t *interface);
129115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
130115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
131115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
132115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
133115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
134115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
135154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
136154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
137154Sigor@sysoev.ru     nxt_work_handler_t handler);
138313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
139313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
140139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
141139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
14253Sigor@sysoev.ru 
14353Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
14453Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
14553Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
14653Sigor@sysoev.ru     nxt_event_engine_t *engine);
147343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
148133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
14953Sigor@sysoev.ru 
150315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
151315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
152315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
153315Sigor@sysoev.ru     nxt_work_t *jobs);
15453Sigor@sysoev.ru 
15553Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
1561545Smax.romanov@nginx.com static void nxt_router_rt_add_port(nxt_task_t *task, void *obj,
1571545Smax.romanov@nginx.com     void *data);
15853Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
15953Sigor@sysoev.ru     void *data);
16053Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
16153Sigor@sysoev.ru     void *data);
16253Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
16353Sigor@sysoev.ru     void *data);
164313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
165313Sigor@sysoev.ru     void *data);
16653Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
16753Sigor@sysoev.ru     void *data);
16853Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
16953Sigor@sysoev.ru     void *data);
1701547Smax.romanov@nginx.com static void nxt_router_req_headers_ack_handler(nxt_task_t *task,
1711547Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, nxt_request_rpc_data_t *req_rpc_data);
172359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
173359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
17453Sigor@sysoev.ru 
175630Svbart@nginx.com static void nxt_router_access_log_writer(nxt_task_t *task,
176630Svbart@nginx.com     nxt_http_request_t *r, nxt_router_access_log_t *access_log);
177630Svbart@nginx.com static u_char *nxt_router_access_log_date(u_char *buf, nxt_realtime_t *now,
178630Svbart@nginx.com     struct tm *tm, size_t size, const char *format);
179630Svbart@nginx.com static void nxt_router_access_log_open(nxt_task_t *task,
180630Svbart@nginx.com     nxt_router_temp_conf_t *tmcf);
181630Svbart@nginx.com static void nxt_router_access_log_ready(nxt_task_t *task,
182630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
183630Svbart@nginx.com static void nxt_router_access_log_error(nxt_task_t *task,
184630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
185630Svbart@nginx.com static void nxt_router_access_log_release(nxt_task_t *task,
186630Svbart@nginx.com     nxt_thread_spinlock_t *lock, nxt_router_access_log_t *access_log);
187651Svbart@nginx.com static void nxt_router_access_log_reopen_completion(nxt_task_t *task, void *obj,
188651Svbart@nginx.com     void *data);
189631Svbart@nginx.com static void nxt_router_access_log_reopen_ready(nxt_task_t *task,
190631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
191631Svbart@nginx.com static void nxt_router_access_log_reopen_error(nxt_task_t *task,
192631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
193630Svbart@nginx.com 
194343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
195343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
1961547Smax.romanov@nginx.com static nxt_int_t nxt_router_app_shared_port_send(nxt_task_t *task,
1971547Smax.romanov@nginx.com     nxt_port_t *app_port);
198343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
199343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
200343Smax.romanov@nginx.com 
201753Smax.romanov@nginx.com static void nxt_router_app_unlink(nxt_task_t *task, nxt_app_t *app);
2021123Smax.romanov@nginx.com 
203343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
2041123Smax.romanov@nginx.com     nxt_apr_action_t action);
2051547Smax.romanov@nginx.com static void nxt_router_app_port_get(nxt_task_t *task, nxt_app_t *app,
2061547Smax.romanov@nginx.com     nxt_request_rpc_data_t *req_rpc_data);
207*1561Smax.romanov@nginx.com static void nxt_router_http_request_error(nxt_task_t *task, void *obj,
208*1561Smax.romanov@nginx.com     void *data);
2091547Smax.romanov@nginx.com static void nxt_router_http_request_done(nxt_task_t *task, void *obj,
2101547Smax.romanov@nginx.com     void *data);
211141Smax.romanov@nginx.com 
212425Smax.romanov@nginx.com static void nxt_router_app_prepare_request(nxt_task_t *task,
2131547Smax.romanov@nginx.com     nxt_request_rpc_data_t *req_rpc_data);
2141007Salexander.borisov@nginx.com static nxt_buf_t *nxt_router_prepare_msg(nxt_task_t *task,
2151547Smax.romanov@nginx.com     nxt_http_request_t *r, nxt_app_t *app, const nxt_str_t *prefix);
216510Salexander.borisov@nginx.com 
217318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
218507Smax.romanov@nginx.com static void nxt_router_adjust_idle_timer(nxt_task_t *task, void *obj,
219507Smax.romanov@nginx.com     void *data);
220507Smax.romanov@nginx.com static void nxt_router_app_idle_timeout(nxt_task_t *task, void *obj,
221507Smax.romanov@nginx.com     void *data);
222753Smax.romanov@nginx.com static void nxt_router_app_joint_release_handler(nxt_task_t *task, void *obj,
223507Smax.romanov@nginx.com     void *data);
224753Smax.romanov@nginx.com static void nxt_router_free_app(nxt_task_t *task, void *obj, void *data);
225431Sigor@sysoev.ru 
226431Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_request_send_state;
227431Sigor@sysoev.ru static void nxt_http_request_send_body(nxt_task_t *task, void *obj, void *data);
228141Smax.romanov@nginx.com 
229753Smax.romanov@nginx.com static void nxt_router_app_joint_use(nxt_task_t *task,
230753Smax.romanov@nginx.com     nxt_app_joint_t *app_joint, int i);
231753Smax.romanov@nginx.com 
2321547Smax.romanov@nginx.com static void nxt_router_http_request_release_post(nxt_task_t *task,
2331007Salexander.borisov@nginx.com     nxt_http_request_t *r);
2341007Salexander.borisov@nginx.com static void nxt_router_http_request_release(nxt_task_t *task, void *obj,
2351007Salexander.borisov@nginx.com     void *data);
2361321Smax.romanov@nginx.com static void nxt_router_oosm_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg);
2371545Smax.romanov@nginx.com static void nxt_router_get_port_handler(nxt_task_t *task,
2381545Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
2391546Smax.romanov@nginx.com static void nxt_router_get_mmap_handler(nxt_task_t *task,
2401546Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg);
2411007Salexander.borisov@nginx.com 
2421149Smax.romanov@nginx.com extern const nxt_http_request_state_t  nxt_http_websocket;
2431131Smax.romanov@nginx.com 
244119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
24520Sigor@sysoev.ru 
246743Smax.romanov@nginx.com static const nxt_str_t http_prefix = nxt_string("HTTP_");
247743Smax.romanov@nginx.com static const nxt_str_t empty_prefix = nxt_string("");
248743Smax.romanov@nginx.com 
249743Smax.romanov@nginx.com static const nxt_str_t  *nxt_app_msg_prefix[] = {
250804Svbart@nginx.com     &empty_prefix,
251743Smax.romanov@nginx.com     &http_prefix,
252743Smax.romanov@nginx.com     &http_prefix,
253743Smax.romanov@nginx.com     &http_prefix,
254743Smax.romanov@nginx.com     &http_prefix,
255977Smax.romanov@gmail.com     &empty_prefix,
256216Sigor@sysoev.ru };
257216Sigor@sysoev.ru 
258216Sigor@sysoev.ru 
2591488St.nateldemoura@f5.com static const nxt_port_handlers_t  nxt_router_process_port_handlers = {
2601488St.nateldemoura@f5.com     .quit         = nxt_signal_quit_handler,
261662Smax.romanov@nginx.com     .new_port     = nxt_router_new_port_handler,
2621545Smax.romanov@nginx.com     .get_port     = nxt_router_get_port_handler,
263662Smax.romanov@nginx.com     .change_file  = nxt_port_change_log_file_handler,
264662Smax.romanov@nginx.com     .mmap         = nxt_port_mmap_handler,
2651546Smax.romanov@nginx.com     .get_mmap     = nxt_router_get_mmap_handler,
266662Smax.romanov@nginx.com     .data         = nxt_router_conf_data_handler,
267662Smax.romanov@nginx.com     .remove_pid   = nxt_router_remove_pid_handler,
268662Smax.romanov@nginx.com     .access_log   = nxt_router_access_log_reopen_handler,
269662Smax.romanov@nginx.com     .rpc_ready    = nxt_port_rpc_handler,
270662Smax.romanov@nginx.com     .rpc_error    = nxt_port_rpc_handler,
2711321Smax.romanov@nginx.com     .oosm         = nxt_router_oosm_handler,
272662Smax.romanov@nginx.com };
273662Smax.romanov@nginx.com 
274662Smax.romanov@nginx.com 
2751488St.nateldemoura@f5.com const nxt_process_init_t  nxt_router_process = {
2761488St.nateldemoura@f5.com     .name           = "router",
2771488St.nateldemoura@f5.com     .type           = NXT_PROCESS_ROUTER,
2781488St.nateldemoura@f5.com     .prefork        = nxt_router_prefork,
2791488St.nateldemoura@f5.com     .restart        = 1,
2801488St.nateldemoura@f5.com     .setup          = nxt_process_core_setup,
2811488St.nateldemoura@f5.com     .start          = nxt_router_start,
2821488St.nateldemoura@f5.com     .port_handlers  = &nxt_router_process_port_handlers,
2831488St.nateldemoura@f5.com     .signals        = nxt_process_signals,
2841488St.nateldemoura@f5.com };
2851488St.nateldemoura@f5.com 
2861488St.nateldemoura@f5.com 
2871509Sigor@sysoev.ru /* Queues of nxt_socket_conf_t */
2881509Sigor@sysoev.ru nxt_queue_t  creating_sockets;
2891509Sigor@sysoev.ru nxt_queue_t  pending_sockets;
2901509Sigor@sysoev.ru nxt_queue_t  updating_sockets;
2911509Sigor@sysoev.ru nxt_queue_t  keeping_sockets;
2921509Sigor@sysoev.ru nxt_queue_t  deleting_sockets;
2931509Sigor@sysoev.ru 
2941509Sigor@sysoev.ru 
2951488St.nateldemoura@f5.com static nxt_int_t
2961488St.nateldemoura@f5.com nxt_router_prefork(nxt_task_t *task, nxt_process_t *process, nxt_mp_t *mp)
2971488St.nateldemoura@f5.com {
2981488St.nateldemoura@f5.com     nxt_runtime_stop_app_processes(task, task->thread->runtime);
2991488St.nateldemoura@f5.com 
3001488St.nateldemoura@f5.com     return NXT_OK;
3011488St.nateldemoura@f5.com }
3021488St.nateldemoura@f5.com 
3031488St.nateldemoura@f5.com 
3041488St.nateldemoura@f5.com static nxt_int_t
3051488St.nateldemoura@f5.com nxt_router_start(nxt_task_t *task, nxt_process_data_t *data)
30620Sigor@sysoev.ru {
307141Smax.romanov@nginx.com     nxt_int_t      ret;
308662Smax.romanov@nginx.com     nxt_port_t     *controller_port;
309141Smax.romanov@nginx.com     nxt_router_t   *router;
310141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
311141Smax.romanov@nginx.com 
312141Smax.romanov@nginx.com     rt = task->thread->runtime;
31353Sigor@sysoev.ru 
3141488St.nateldemoura@f5.com     nxt_log(task, NXT_LOG_INFO, "router started");
3151488St.nateldemoura@f5.com 
316771Sigor@sysoev.ru #if (NXT_TLS)
317771Sigor@sysoev.ru     rt->tls = nxt_service_get(rt->services, "SSL/TLS", "OpenSSL");
318771Sigor@sysoev.ru     if (nxt_slow_path(rt->tls == NULL)) {
319771Sigor@sysoev.ru         return NXT_ERROR;
320771Sigor@sysoev.ru     }
321771Sigor@sysoev.ru 
322771Sigor@sysoev.ru     ret = rt->tls->library_init(task);
323771Sigor@sysoev.ru     if (nxt_slow_path(ret != NXT_OK)) {
324771Sigor@sysoev.ru         return ret;
325771Sigor@sysoev.ru     }
326771Sigor@sysoev.ru #endif
327771Sigor@sysoev.ru 
3281459Smax.romanov@nginx.com     ret = nxt_http_init(task);
32988Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
33088Smax.romanov@nginx.com         return ret;
33188Smax.romanov@nginx.com     }
33288Smax.romanov@nginx.com 
33353Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
33453Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
33553Sigor@sysoev.ru         return NXT_ERROR;
33653Sigor@sysoev.ru     }
33753Sigor@sysoev.ru 
33853Sigor@sysoev.ru     nxt_queue_init(&router->engines);
33953Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
340133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
34153Sigor@sysoev.ru 
342119Smax.romanov@nginx.com     nxt_router = router;
343119Smax.romanov@nginx.com 
344662Smax.romanov@nginx.com     controller_port = rt->port_by_type[NXT_PROCESS_CONTROLLER];
345662Smax.romanov@nginx.com     if (controller_port != NULL) {
346662Smax.romanov@nginx.com         nxt_router_greet_controller(task, controller_port);
347662Smax.romanov@nginx.com     }
348662Smax.romanov@nginx.com 
349115Sigor@sysoev.ru     return NXT_OK;
350115Sigor@sysoev.ru }
351115Sigor@sysoev.ru 
352115Sigor@sysoev.ru 
353343Smax.romanov@nginx.com static void
354662Smax.romanov@nginx.com nxt_router_greet_controller(nxt_task_t *task, nxt_port_t *controller_port)
355662Smax.romanov@nginx.com {
356662Smax.romanov@nginx.com     nxt_port_socket_write(task, controller_port, NXT_PORT_MSG_PROCESS_READY,
357662Smax.romanov@nginx.com                           -1, 0, 0, NULL);
358662Smax.romanov@nginx.com }
359662Smax.romanov@nginx.com 
360662Smax.romanov@nginx.com 
361662Smax.romanov@nginx.com static void
362507Smax.romanov@nginx.com nxt_router_start_app_process_handler(nxt_task_t *task, nxt_port_t *port,
363507Smax.romanov@nginx.com     void *data)
364167Smax.romanov@nginx.com {
365343Smax.romanov@nginx.com     size_t         size;
366343Smax.romanov@nginx.com     uint32_t       stream;
367430Sigor@sysoev.ru     nxt_mp_t       *mp;
368648Svbart@nginx.com     nxt_int_t      ret;
369343Smax.romanov@nginx.com     nxt_app_t      *app;
370343Smax.romanov@nginx.com     nxt_buf_t      *b;
371343Smax.romanov@nginx.com     nxt_port_t     *main_port;
372343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
373343Smax.romanov@nginx.com 
374343Smax.romanov@nginx.com     app = data;
375167Smax.romanov@nginx.com 
376167Smax.romanov@nginx.com     rt = task->thread->runtime;
377240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
378167Smax.romanov@nginx.com 
379507Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start process", &app->name, app);
380343Smax.romanov@nginx.com 
381343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
382343Smax.romanov@nginx.com 
383343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
384343Smax.romanov@nginx.com 
385343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
386343Smax.romanov@nginx.com         goto failed;
387167Smax.romanov@nginx.com     }
388167Smax.romanov@nginx.com 
389343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
390343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
391343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
392343Smax.romanov@nginx.com 
393753Smax.romanov@nginx.com     nxt_router_app_joint_use(task, app->joint, 1);
394753Smax.romanov@nginx.com 
395343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
396343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
397343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
398753Smax.romanov@nginx.com                                            -1, app->joint);
399343Smax.romanov@nginx.com 
400343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
401753Smax.romanov@nginx.com         nxt_router_app_joint_use(task, app->joint, -1);
402753Smax.romanov@nginx.com 
403343Smax.romanov@nginx.com         goto failed;
404343Smax.romanov@nginx.com     }
405343Smax.romanov@nginx.com 
4061488St.nateldemoura@f5.com     ret = nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_PROCESS,
4071488St.nateldemoura@f5.com                                 -1, stream, port->id, b);
408648Svbart@nginx.com 
409648Svbart@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
410648Svbart@nginx.com         nxt_port_rpc_cancel(task, port, stream);
411753Smax.romanov@nginx.com 
412753Smax.romanov@nginx.com         nxt_router_app_joint_use(task, app->joint, -1);
413753Smax.romanov@nginx.com 
414648Svbart@nginx.com         goto failed;
415648Svbart@nginx.com     }
416343Smax.romanov@nginx.com 
417753Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
418753Smax.romanov@nginx.com 
419343Smax.romanov@nginx.com     return;
420343Smax.romanov@nginx.com 
421343Smax.romanov@nginx.com failed:
422343Smax.romanov@nginx.com 
423648Svbart@nginx.com     if (b != NULL) {
424648Svbart@nginx.com         mp = b->data;
425648Svbart@nginx.com         nxt_mp_free(mp, b);
426648Svbart@nginx.com         nxt_mp_release(mp);
427648Svbart@nginx.com     }
428648Svbart@nginx.com 
429343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
430343Smax.romanov@nginx.com 
431507Smax.romanov@nginx.com     app->pending_processes--;
432343Smax.romanov@nginx.com 
433343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
434343Smax.romanov@nginx.com 
435343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
436167Smax.romanov@nginx.com }
437167Smax.romanov@nginx.com 
438167Smax.romanov@nginx.com 
439753Smax.romanov@nginx.com static void
440753Smax.romanov@nginx.com nxt_router_app_joint_use(nxt_task_t *task, nxt_app_joint_t *app_joint, int i)
441753Smax.romanov@nginx.com {
442753Smax.romanov@nginx.com     app_joint->use_count += i;
443753Smax.romanov@nginx.com 
444753Smax.romanov@nginx.com     if (app_joint->use_count == 0) {
445753Smax.romanov@nginx.com         nxt_assert(app_joint->app == NULL);
446753Smax.romanov@nginx.com 
447753Smax.romanov@nginx.com         nxt_free(app_joint);
448753Smax.romanov@nginx.com     }
449753Smax.romanov@nginx.com }
450753Smax.romanov@nginx.com 
451753Smax.romanov@nginx.com 
452343Smax.romanov@nginx.com static nxt_int_t
453507Smax.romanov@nginx.com nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app)
454141Smax.romanov@nginx.com {
455343Smax.romanov@nginx.com     nxt_int_t      res;
456343Smax.romanov@nginx.com     nxt_port_t     *router_port;
457343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
458343Smax.romanov@nginx.com 
4591549Smax.romanov@nginx.com     nxt_debug(task, "app '%V' start process", &app->name);
4601549Smax.romanov@nginx.com 
461343Smax.romanov@nginx.com     rt = task->thread->runtime;
462343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
463343Smax.romanov@nginx.com 
464343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
465343Smax.romanov@nginx.com 
466507Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_app_process_handler,
467343Smax.romanov@nginx.com                         app);
468343Smax.romanov@nginx.com 
469343Smax.romanov@nginx.com     if (res == NXT_OK) {
470343Smax.romanov@nginx.com         return res;
471318Smax.romanov@nginx.com     }
472318Smax.romanov@nginx.com 
473343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
474343Smax.romanov@nginx.com 
475507Smax.romanov@nginx.com     app->pending_processes--;
476343Smax.romanov@nginx.com 
477343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
478343Smax.romanov@nginx.com 
479343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
480343Smax.romanov@nginx.com 
481343Smax.romanov@nginx.com     return NXT_ERROR;
482318Smax.romanov@nginx.com }
483318Smax.romanov@nginx.com 
484318Smax.romanov@nginx.com 
485423Smax.romanov@nginx.com nxt_inline nxt_bool_t
4861555Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_request_rpc_data_t *req_rpc_data)
487423Smax.romanov@nginx.com {
4881555Smax.romanov@nginx.com     nxt_buf_t       *b, *next;
4891555Smax.romanov@nginx.com     nxt_bool_t      cancelled;
4901555Smax.romanov@nginx.com     nxt_msg_info_t  *msg_info;
4911555Smax.romanov@nginx.com 
4921555Smax.romanov@nginx.com     msg_info = &req_rpc_data->msg_info;
493423Smax.romanov@nginx.com 
494423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
495423Smax.romanov@nginx.com         return 0;
496423Smax.romanov@nginx.com     }
497423Smax.romanov@nginx.com 
4981555Smax.romanov@nginx.com     cancelled = nxt_app_queue_cancel(req_rpc_data->app->shared_port->queue,
4991555Smax.romanov@nginx.com                                      msg_info->tracking_cookie,
5001555Smax.romanov@nginx.com                                      req_rpc_data->stream);
501423Smax.romanov@nginx.com 
502423Smax.romanov@nginx.com     if (cancelled) {
5031555Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router",
5041555Smax.romanov@nginx.com                   req_rpc_data->stream);
505423Smax.romanov@nginx.com     }
506423Smax.romanov@nginx.com 
507423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
508423Smax.romanov@nginx.com         next = b->next;
5091269Sigor@sysoev.ru         b->next = NULL;
510423Smax.romanov@nginx.com 
511423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
512423Smax.romanov@nginx.com 
513423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
514423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
515423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
516423Smax.romanov@nginx.com         }
517423Smax.romanov@nginx.com     }
518423Smax.romanov@nginx.com 
519423Smax.romanov@nginx.com     msg_info->buf = NULL;
520423Smax.romanov@nginx.com 
521423Smax.romanov@nginx.com     return cancelled;
522423Smax.romanov@nginx.com }
523423Smax.romanov@nginx.com 
524423Smax.romanov@nginx.com 
525425Smax.romanov@nginx.com nxt_inline nxt_bool_t
526425Smax.romanov@nginx.com nxt_queue_chk_remove(nxt_queue_link_t *lnk)
527425Smax.romanov@nginx.com {
528425Smax.romanov@nginx.com     if (lnk->next != NULL) {
529425Smax.romanov@nginx.com         nxt_queue_remove(lnk);
530425Smax.romanov@nginx.com 
531425Smax.romanov@nginx.com         lnk->next = NULL;
532425Smax.romanov@nginx.com 
533425Smax.romanov@nginx.com         return 1;
534425Smax.romanov@nginx.com     }
535425Smax.romanov@nginx.com 
536425Smax.romanov@nginx.com     return 0;
537425Smax.romanov@nginx.com }
538425Smax.romanov@nginx.com 
539425Smax.romanov@nginx.com 
540343Smax.romanov@nginx.com nxt_inline void
5411123Smax.romanov@nginx.com nxt_request_rpc_data_unlink(nxt_task_t *task,
5421123Smax.romanov@nginx.com     nxt_request_rpc_data_t *req_rpc_data)
543343Smax.romanov@nginx.com {
544*1561Smax.romanov@nginx.com     nxt_app_t           *app;
545*1561Smax.romanov@nginx.com     nxt_bool_t          unlinked;
5461547Smax.romanov@nginx.com     nxt_http_request_t  *r;
5471547Smax.romanov@nginx.com 
5481555Smax.romanov@nginx.com     nxt_router_msg_cancel(task, req_rpc_data);
5491123Smax.romanov@nginx.com 
5501123Smax.romanov@nginx.com     if (req_rpc_data->app_port != NULL) {
5511123Smax.romanov@nginx.com         nxt_router_app_port_release(task, req_rpc_data->app_port,
5521123Smax.romanov@nginx.com                                     req_rpc_data->apr_action);
5531123Smax.romanov@nginx.com 
5541123Smax.romanov@nginx.com         req_rpc_data->app_port = NULL;
5551123Smax.romanov@nginx.com     }
5561123Smax.romanov@nginx.com 
557*1561Smax.romanov@nginx.com     app = req_rpc_data->app;
5581547Smax.romanov@nginx.com     r = req_rpc_data->request;
5591547Smax.romanov@nginx.com 
5601547Smax.romanov@nginx.com     if (r != NULL) {
5611547Smax.romanov@nginx.com         r->timer_data = NULL;
5621547Smax.romanov@nginx.com 
5631547Smax.romanov@nginx.com         nxt_router_http_request_release_post(task, r);
5641547Smax.romanov@nginx.com 
5651547Smax.romanov@nginx.com         r->req_rpc_data = NULL;
5661123Smax.romanov@nginx.com         req_rpc_data->request = NULL;
567*1561Smax.romanov@nginx.com 
568*1561Smax.romanov@nginx.com         if (app != NULL) {
569*1561Smax.romanov@nginx.com             unlinked = 0;
570*1561Smax.romanov@nginx.com 
571*1561Smax.romanov@nginx.com             nxt_thread_mutex_lock(&app->mutex);
572*1561Smax.romanov@nginx.com 
573*1561Smax.romanov@nginx.com             if (r->app_link.next != NULL) {
574*1561Smax.romanov@nginx.com                 nxt_queue_remove(&r->app_link);
575*1561Smax.romanov@nginx.com                 r->app_link.next = NULL;
576*1561Smax.romanov@nginx.com 
577*1561Smax.romanov@nginx.com                 unlinked = 1;
578*1561Smax.romanov@nginx.com             }
579*1561Smax.romanov@nginx.com 
580*1561Smax.romanov@nginx.com             nxt_thread_mutex_unlock(&app->mutex);
581*1561Smax.romanov@nginx.com 
582*1561Smax.romanov@nginx.com             if (unlinked) {
583*1561Smax.romanov@nginx.com                 nxt_mp_release(r->mem_pool);
584*1561Smax.romanov@nginx.com             }
585*1561Smax.romanov@nginx.com         }
586*1561Smax.romanov@nginx.com     }
587*1561Smax.romanov@nginx.com 
588*1561Smax.romanov@nginx.com     if (app != NULL) {
589*1561Smax.romanov@nginx.com         nxt_router_app_use(task, app, -1);
590*1561Smax.romanov@nginx.com 
591*1561Smax.romanov@nginx.com         req_rpc_data->app = NULL;
592346Smax.romanov@nginx.com     }
5931547Smax.romanov@nginx.com 
5941547Smax.romanov@nginx.com     if (req_rpc_data->msg_info.body_fd != -1) {
5951547Smax.romanov@nginx.com         nxt_fd_close(req_rpc_data->msg_info.body_fd);
5961547Smax.romanov@nginx.com 
5971547Smax.romanov@nginx.com         req_rpc_data->msg_info.body_fd = -1;
5981547Smax.romanov@nginx.com     }
5991547Smax.romanov@nginx.com 
6001547Smax.romanov@nginx.com     if (req_rpc_data->rpc_cancel) {
6011547Smax.romanov@nginx.com         req_rpc_data->rpc_cancel = 0;
6021547Smax.romanov@nginx.com 
6031547Smax.romanov@nginx.com         nxt_port_rpc_cancel(task, task->thread->engine->port,
6041547Smax.romanov@nginx.com                             req_rpc_data->stream);
6051547Smax.romanov@nginx.com     }
606343Smax.romanov@nginx.com }
607343Smax.romanov@nginx.com 
608343Smax.romanov@nginx.com 
6091552Smax.romanov@nginx.com static void
610141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
611141Smax.romanov@nginx.com {
6121555Smax.romanov@nginx.com     nxt_int_t      res;
6131547Smax.romanov@nginx.com     nxt_app_t      *app;
6141547Smax.romanov@nginx.com     nxt_port_t     *port, *main_app_port;
6151547Smax.romanov@nginx.com     nxt_runtime_t  *rt;
6161547Smax.romanov@nginx.com 
617141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
618141Smax.romanov@nginx.com 
6191547Smax.romanov@nginx.com     port = msg->u.new_port;
6201547Smax.romanov@nginx.com 
6211547Smax.romanov@nginx.com     if (port != NULL && port->type == NXT_PROCESS_CONTROLLER) {
622662Smax.romanov@nginx.com         nxt_router_greet_controller(task, msg->u.new_port);
623662Smax.romanov@nginx.com     }
624662Smax.romanov@nginx.com 
6251547Smax.romanov@nginx.com     if (port == NULL || port->type != NXT_PROCESS_APP) {
6261547Smax.romanov@nginx.com 
6271547Smax.romanov@nginx.com         if (msg->port_msg.stream == 0) {
6281547Smax.romanov@nginx.com             return;
6291547Smax.romanov@nginx.com         }
6301547Smax.romanov@nginx.com 
6311547Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
6321555Smax.romanov@nginx.com 
6331555Smax.romanov@nginx.com     } else {
6341558Smax.romanov@nginx.com         if (msg->fd[1] != -1) {
6351558Smax.romanov@nginx.com             res = nxt_router_port_queue_map(task, port, msg->fd[1]);
6361555Smax.romanov@nginx.com             if (nxt_slow_path(res != NXT_OK)) {
6371555Smax.romanov@nginx.com                 return;
6381555Smax.romanov@nginx.com             }
6391555Smax.romanov@nginx.com 
6401558Smax.romanov@nginx.com             nxt_fd_close(msg->fd[1]);
6411558Smax.romanov@nginx.com             msg->fd[1] = -1;
6421555Smax.romanov@nginx.com         }
6431547Smax.romanov@nginx.com     }
6441547Smax.romanov@nginx.com 
6451547Smax.romanov@nginx.com     if (msg->port_msg.stream != 0) {
6461547Smax.romanov@nginx.com         nxt_port_rpc_handler(task, msg);
647141Smax.romanov@nginx.com         return;
648141Smax.romanov@nginx.com     }
649141Smax.romanov@nginx.com 
6501547Smax.romanov@nginx.com     /*
6511547Smax.romanov@nginx.com      * Port with "id == 0" is application 'main' port and it always
6521547Smax.romanov@nginx.com      * should come with non-zero stream.
6531547Smax.romanov@nginx.com      */
6541547Smax.romanov@nginx.com     nxt_assert(port->id != 0);
6551547Smax.romanov@nginx.com 
6561547Smax.romanov@nginx.com     /* Find 'main' app port and get app reference. */
6571547Smax.romanov@nginx.com     rt = task->thread->runtime;
6581547Smax.romanov@nginx.com 
6591547Smax.romanov@nginx.com     /*
6601547Smax.romanov@nginx.com      * It is safe to access 'runtime->ports' hash because 'NEW_PORT'
6611547Smax.romanov@nginx.com      * sent to main port (with id == 0) and processed in main thread.
6621547Smax.romanov@nginx.com      */
6631547Smax.romanov@nginx.com     main_app_port = nxt_port_hash_find(&rt->ports, port->pid, 0);
6641547Smax.romanov@nginx.com     nxt_assert(main_app_port != NULL);
6651547Smax.romanov@nginx.com 
6661547Smax.romanov@nginx.com     app = main_app_port->app;
6671547Smax.romanov@nginx.com     nxt_assert(app != NULL);
6681547Smax.romanov@nginx.com 
6691547Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
6701547Smax.romanov@nginx.com 
6711547Smax.romanov@nginx.com     /* TODO here should be find-and-add code because there can be
6721547Smax.romanov@nginx.com        port waiters in port_hash */
6731547Smax.romanov@nginx.com     nxt_port_hash_add(&app->port_hash, port);
6741547Smax.romanov@nginx.com     app->port_hash_count++;
6751547Smax.romanov@nginx.com 
6761547Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
6771547Smax.romanov@nginx.com 
6781547Smax.romanov@nginx.com     port->app = app;
6791547Smax.romanov@nginx.com     port->main_app_port = main_app_port;
680141Smax.romanov@nginx.com }
681141Smax.romanov@nginx.com 
682141Smax.romanov@nginx.com 
6831552Smax.romanov@nginx.com static void
684139Sigor@sysoev.ru nxt_router_conf_data_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
685115Sigor@sysoev.ru {
6861526Smax.romanov@nginx.com     void                    *p;
6871526Smax.romanov@nginx.com     size_t                  size;
688198Sigor@sysoev.ru     nxt_int_t               ret;
689139Sigor@sysoev.ru     nxt_router_temp_conf_t  *tmcf;
690139Sigor@sysoev.ru 
691139Sigor@sysoev.ru     tmcf = nxt_router_temp_conf(task);
692139Sigor@sysoev.ru     if (nxt_slow_path(tmcf == NULL)) {
693139Sigor@sysoev.ru         return;
69453Sigor@sysoev.ru     }
69553Sigor@sysoev.ru 
6961558Smax.romanov@nginx.com     if (nxt_slow_path(msg->fd[0] == -1)) {
6971526Smax.romanov@nginx.com         nxt_alert(task, "conf_data_handler: invalid file shm fd");
6981526Smax.romanov@nginx.com         return;
6991526Smax.romanov@nginx.com     }
7001526Smax.romanov@nginx.com 
7011526Smax.romanov@nginx.com     if (nxt_buf_mem_used_size(&msg->buf->mem) != sizeof(size_t)) {
7021526Smax.romanov@nginx.com         nxt_alert(task, "conf_data_handler: unexpected buffer size (%d)",
7031526Smax.romanov@nginx.com                   (int) nxt_buf_mem_used_size(&msg->buf->mem));
7041526Smax.romanov@nginx.com 
7051558Smax.romanov@nginx.com         nxt_fd_close(msg->fd[0]);
7061558Smax.romanov@nginx.com         msg->fd[0] = -1;
7071526Smax.romanov@nginx.com 
7081526Smax.romanov@nginx.com         return;
7091526Smax.romanov@nginx.com     }
7101526Smax.romanov@nginx.com 
7111526Smax.romanov@nginx.com     nxt_memcpy(&size, msg->buf->mem.pos, sizeof(size_t));
7121526Smax.romanov@nginx.com 
7131558Smax.romanov@nginx.com     p = nxt_mem_mmap(NULL, size, PROT_READ, MAP_SHARED, msg->fd[0], 0);
7141558Smax.romanov@nginx.com 
7151558Smax.romanov@nginx.com     nxt_fd_close(msg->fd[0]);
7161558Smax.romanov@nginx.com     msg->fd[0] = -1;
7171526Smax.romanov@nginx.com 
7181526Smax.romanov@nginx.com     if (nxt_slow_path(p == MAP_FAILED)) {
7191526Smax.romanov@nginx.com         return;
7201526Smax.romanov@nginx.com     }
7211526Smax.romanov@nginx.com 
7221526Smax.romanov@nginx.com     nxt_debug(task, "conf_data_handler(%uz): %*s", size, size, p);
723423Smax.romanov@nginx.com 
724591Sigor@sysoev.ru     tmcf->router_conf->router = nxt_router;
725139Sigor@sysoev.ru     tmcf->stream = msg->port_msg.stream;
726139Sigor@sysoev.ru     tmcf->port = nxt_runtime_port_find(task->thread->runtime,
727198Sigor@sysoev.ru                                        msg->port_msg.pid,
728198Sigor@sysoev.ru                                        msg->port_msg.reply_port);
729198Sigor@sysoev.ru&