xref: /unit/src/nxt_router.c (revision 662)
120Sigor@sysoev.ru 
220Sigor@sysoev.ru /*
320Sigor@sysoev.ru  * Copyright (C) Igor Sysoev
420Sigor@sysoev.ru  * Copyright (C) Valentin V. Bartenev
520Sigor@sysoev.ru  * Copyright (C) NGINX, Inc.
620Sigor@sysoev.ru  */
720Sigor@sysoev.ru 
853Sigor@sysoev.ru #include <nxt_router.h>
9115Sigor@sysoev.ru #include <nxt_conf.h>
10431Sigor@sysoev.ru #include <nxt_http.h>
1120Sigor@sysoev.ru 
1220Sigor@sysoev.ru 
13115Sigor@sysoev.ru typedef struct {
14318Smax.romanov@nginx.com     nxt_str_t         type;
15507Smax.romanov@nginx.com     uint32_t          processes;
16507Smax.romanov@nginx.com     uint32_t          max_processes;
17507Smax.romanov@nginx.com     uint32_t          spare_processes;
18318Smax.romanov@nginx.com     nxt_msec_t        timeout;
19427Smax.romanov@nginx.com     nxt_msec_t        res_timeout;
20507Smax.romanov@nginx.com     nxt_msec_t        idle_timeout;
21318Smax.romanov@nginx.com     uint32_t          requests;
22318Smax.romanov@nginx.com     nxt_conf_value_t  *limits_value;
23507Smax.romanov@nginx.com     nxt_conf_value_t  *processes_value;
24133Sigor@sysoev.ru } nxt_router_app_conf_t;
25133Sigor@sysoev.ru 
26133Sigor@sysoev.ru 
27133Sigor@sysoev.ru typedef struct {
28133Sigor@sysoev.ru     nxt_str_t  application;
29115Sigor@sysoev.ru } nxt_router_listener_conf_t;
30115Sigor@sysoev.ru 
31115Sigor@sysoev.ru 
32423Smax.romanov@nginx.com typedef struct nxt_msg_info_s {
33423Smax.romanov@nginx.com     nxt_buf_t                 *buf;
34423Smax.romanov@nginx.com     nxt_port_mmap_tracking_t  tracking;
35423Smax.romanov@nginx.com     nxt_work_handler_t        completion_handler;
36423Smax.romanov@nginx.com } nxt_msg_info_t;
37423Smax.romanov@nginx.com 
38423Smax.romanov@nginx.com 
39167Smax.romanov@nginx.com typedef struct nxt_req_app_link_s nxt_req_app_link_t;
40141Smax.romanov@nginx.com 
41141Smax.romanov@nginx.com 
42318Smax.romanov@nginx.com typedef struct {
43431Sigor@sysoev.ru     uint32_t                 stream;
44431Sigor@sysoev.ru     nxt_app_t                *app;
45431Sigor@sysoev.ru     nxt_port_t               *app_port;
46431Sigor@sysoev.ru     nxt_app_parse_ctx_t      *ap;
47431Sigor@sysoev.ru     nxt_msg_info_t           msg_info;
48431Sigor@sysoev.ru     nxt_req_app_link_t       *ra;
49431Sigor@sysoev.ru 
50431Sigor@sysoev.ru     nxt_queue_link_t         link;     /* for nxt_conn_t.requests */
51318Smax.romanov@nginx.com } nxt_req_conn_link_t;
52318Smax.romanov@nginx.com 
53318Smax.romanov@nginx.com 
54167Smax.romanov@nginx.com struct nxt_req_app_link_s {
55318Smax.romanov@nginx.com     uint32_t             stream;
56425Smax.romanov@nginx.com     nxt_atomic_t         use_count;
57167Smax.romanov@nginx.com     nxt_port_t           *app_port;
58167Smax.romanov@nginx.com     nxt_port_t           *reply_port;
59167Smax.romanov@nginx.com     nxt_app_parse_ctx_t  *ap;
60423Smax.romanov@nginx.com     nxt_msg_info_t       msg_info;
61167Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
62167Smax.romanov@nginx.com 
63427Smax.romanov@nginx.com     nxt_nsec_t           res_time;
64427Smax.romanov@nginx.com 
65425Smax.romanov@nginx.com     nxt_queue_link_t     link_app_requests; /* for nxt_app_t.requests */
66425Smax.romanov@nginx.com     nxt_queue_link_t     link_port_pending; /* for nxt_port_t.pending_requests */
67427Smax.romanov@nginx.com     nxt_queue_link_t     link_app_pending;  /* for nxt_app_t.pending */
68167Smax.romanov@nginx.com 
69167Smax.romanov@nginx.com     nxt_mp_t             *mem_pool;
70167Smax.romanov@nginx.com     nxt_work_t           work;
71345Smax.romanov@nginx.com 
72345Smax.romanov@nginx.com     int                  err_code;
73345Smax.romanov@nginx.com     const char           *err_str;
74167Smax.romanov@nginx.com };
75167Smax.romanov@nginx.com 
76167Smax.romanov@nginx.com 
77198Sigor@sysoev.ru typedef struct {
78198Sigor@sysoev.ru     nxt_socket_conf_t       *socket_conf;
79198Sigor@sysoev.ru     nxt_router_temp_conf_t  *temp_conf;
80198Sigor@sysoev.ru } nxt_socket_rpc_t;
81198Sigor@sysoev.ru 
82198Sigor@sysoev.ru 
83507Smax.romanov@nginx.com typedef struct {
84507Smax.romanov@nginx.com     nxt_app_t               *app;
85507Smax.romanov@nginx.com     nxt_router_temp_conf_t  *temp_conf;
86507Smax.romanov@nginx.com } nxt_app_rpc_t;
87507Smax.romanov@nginx.com 
88507Smax.romanov@nginx.com 
89427Smax.romanov@nginx.com struct nxt_port_select_state_s {
90427Smax.romanov@nginx.com     nxt_app_t           *app;
91427Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
92427Smax.romanov@nginx.com 
93427Smax.romanov@nginx.com     nxt_port_t          *failed_port;
94427Smax.romanov@nginx.com     int                 failed_port_use_delta;
95427Smax.romanov@nginx.com 
96507Smax.romanov@nginx.com     uint8_t             start_process;    /* 1 bit */
97427Smax.romanov@nginx.com     nxt_req_app_link_t  *shared_ra;
98427Smax.romanov@nginx.com     nxt_port_t          *port;
99427Smax.romanov@nginx.com };
100427Smax.romanov@nginx.com 
101427Smax.romanov@nginx.com typedef struct nxt_port_select_state_s nxt_port_select_state_t;
102427Smax.romanov@nginx.com 
103*662Smax.romanov@nginx.com static void nxt_router_greet_controller(nxt_task_t *task,
104*662Smax.romanov@nginx.com     nxt_port_t *controller_port);
105*662Smax.romanov@nginx.com 
106427Smax.romanov@nginx.com static void nxt_router_port_select(nxt_task_t *task,
107427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
108427Smax.romanov@nginx.com 
109427Smax.romanov@nginx.com static nxt_int_t nxt_router_port_post_select(nxt_task_t *task,
110427Smax.romanov@nginx.com     nxt_port_select_state_t *state);
111427Smax.romanov@nginx.com 
112507Smax.romanov@nginx.com static nxt_int_t nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app);
113343Smax.romanov@nginx.com 
114425Smax.romanov@nginx.com nxt_inline void
115425Smax.romanov@nginx.com nxt_router_ra_inc_use(nxt_req_app_link_t *ra)
116425Smax.romanov@nginx.com {
117425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, 1);
118425Smax.romanov@nginx.com }
119425Smax.romanov@nginx.com 
120425Smax.romanov@nginx.com nxt_inline void
121425Smax.romanov@nginx.com nxt_router_ra_dec_use(nxt_req_app_link_t *ra)
122425Smax.romanov@nginx.com {
123538Svbart@nginx.com #if (NXT_DEBUG)
124425Smax.romanov@nginx.com     int  c;
125425Smax.romanov@nginx.com 
126425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, -1);
127425Smax.romanov@nginx.com 
128425Smax.romanov@nginx.com     nxt_assert(c > 1);
129538Svbart@nginx.com #else
130538Svbart@nginx.com     (void) nxt_atomic_fetch_add(&ra->use_count, -1);
131538Svbart@nginx.com #endif
132425Smax.romanov@nginx.com }
133425Smax.romanov@nginx.com 
134425Smax.romanov@nginx.com static void nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i);
135425Smax.romanov@nginx.com 
136139Sigor@sysoev.ru static nxt_router_temp_conf_t *nxt_router_temp_conf(nxt_task_t *task);
137198Sigor@sysoev.ru static void nxt_router_conf_apply(nxt_task_t *task, void *obj, void *data);
138198Sigor@sysoev.ru static void nxt_router_conf_ready(nxt_task_t *task,
139139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
140139Sigor@sysoev.ru static void nxt_router_conf_error(nxt_task_t *task,
141139Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
142139Sigor@sysoev.ru static void nxt_router_conf_send(nxt_task_t *task,
143193Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_port_msg_type_t type);
14453Sigor@sysoev.ru 
145115Sigor@sysoev.ru static nxt_int_t nxt_router_conf_create(nxt_task_t *task,
146115Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, u_char *start, u_char *end);
147133Sigor@sysoev.ru static nxt_app_t *nxt_router_app_find(nxt_queue_t *queue, nxt_str_t *name);
148133Sigor@sysoev.ru static nxt_app_t *nxt_router_listener_application(nxt_router_temp_conf_t *tmcf,
149133Sigor@sysoev.ru     nxt_str_t *name);
150198Sigor@sysoev.ru static void nxt_router_listen_socket_rpc_create(nxt_task_t *task,
151198Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_socket_conf_t *skcf);
152198Sigor@sysoev.ru static void nxt_router_listen_socket_ready(nxt_task_t *task,
153198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
154198Sigor@sysoev.ru static void nxt_router_listen_socket_error(nxt_task_t *task,
155198Sigor@sysoev.ru     nxt_port_recv_msg_t *msg, void *data);
156507Smax.romanov@nginx.com static void nxt_router_app_rpc_create(nxt_task_t *task,
157507Smax.romanov@nginx.com     nxt_router_temp_conf_t *tmcf, nxt_app_t *app);
158507Smax.romanov@nginx.com static void nxt_router_app_prefork_ready(nxt_task_t *task,
159507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
160507Smax.romanov@nginx.com static void nxt_router_app_prefork_error(nxt_task_t *task,
161507Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
162359Sigor@sysoev.ru static nxt_socket_conf_t *nxt_router_socket_conf(nxt_task_t *task,
163359Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf, nxt_str_t *name);
164359Sigor@sysoev.ru static nxt_int_t nxt_router_listen_socket_find(nxt_router_temp_conf_t *tmcf,
165359Sigor@sysoev.ru     nxt_socket_conf_t *nskcf, nxt_sockaddr_t *sa);
16653Sigor@sysoev.ru 
16753Sigor@sysoev.ru static nxt_int_t nxt_router_engines_create(nxt_task_t *task,
16853Sigor@sysoev.ru     nxt_router_t *router, nxt_router_temp_conf_t *tmcf,
16953Sigor@sysoev.ru     const nxt_event_interface_t *interface);
170115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_create(nxt_router_temp_conf_t *tmcf,
171115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
172115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_update(nxt_router_temp_conf_t *tmcf,
173115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
174115Sigor@sysoev.ru static nxt_int_t nxt_router_engine_conf_delete(nxt_router_temp_conf_t *tmcf,
175115Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
176154Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_create(nxt_router_temp_conf_t *tmcf,
177154Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets,
178154Sigor@sysoev.ru     nxt_work_handler_t handler);
179313Sigor@sysoev.ru static nxt_int_t nxt_router_engine_quit(nxt_router_temp_conf_t *tmcf,
180313Sigor@sysoev.ru     nxt_router_engine_conf_t *recf);
181139Sigor@sysoev.ru static nxt_int_t nxt_router_engine_joints_delete(nxt_router_temp_conf_t *tmcf,
182139Sigor@sysoev.ru     nxt_router_engine_conf_t *recf, nxt_queue_t *sockets);
18353Sigor@sysoev.ru 
18453Sigor@sysoev.ru static nxt_int_t nxt_router_threads_create(nxt_task_t *task, nxt_runtime_t *rt,
18553Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
18653Sigor@sysoev.ru static nxt_int_t nxt_router_thread_create(nxt_task_t *task, nxt_runtime_t *rt,
18753Sigor@sysoev.ru     nxt_event_engine_t *engine);
188343Smax.romanov@nginx.com static void nxt_router_apps_sort(nxt_task_t *task, nxt_router_t *router,
189133Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
19053Sigor@sysoev.ru 
191315Sigor@sysoev.ru static void nxt_router_engines_post(nxt_router_t *router,
192315Sigor@sysoev.ru     nxt_router_temp_conf_t *tmcf);
193315Sigor@sysoev.ru static void nxt_router_engine_post(nxt_event_engine_t *engine,
194315Sigor@sysoev.ru     nxt_work_t *jobs);
19553Sigor@sysoev.ru 
19653Sigor@sysoev.ru static void nxt_router_thread_start(void *data);
19753Sigor@sysoev.ru static void nxt_router_listen_socket_create(nxt_task_t *task, void *obj,
19853Sigor@sysoev.ru     void *data);
19953Sigor@sysoev.ru static void nxt_router_listen_socket_update(nxt_task_t *task, void *obj,
20053Sigor@sysoev.ru     void *data);
20153Sigor@sysoev.ru static void nxt_router_listen_socket_delete(nxt_task_t *task, void *obj,
20253Sigor@sysoev.ru     void *data);
203313Sigor@sysoev.ru static void nxt_router_worker_thread_quit(nxt_task_t *task, void *obj,
204313Sigor@sysoev.ru     void *data);
20553Sigor@sysoev.ru static void nxt_router_listen_socket_close(nxt_task_t *task, void *obj,
20653Sigor@sysoev.ru     void *data);
20753Sigor@sysoev.ru static void nxt_router_thread_exit_handler(nxt_task_t *task, void *obj,
20853Sigor@sysoev.ru     void *data);
209359Sigor@sysoev.ru static void nxt_router_listen_socket_release(nxt_task_t *task,
210359Sigor@sysoev.ru     nxt_socket_conf_t *skcf);
21153Sigor@sysoev.ru static void nxt_router_conf_release(nxt_task_t *task,
21253Sigor@sysoev.ru     nxt_socket_conf_joint_t *joint);
21353Sigor@sysoev.ru 
214630Svbart@nginx.com static void nxt_router_access_log_writer(nxt_task_t *task,
215630Svbart@nginx.com     nxt_http_request_t *r, nxt_router_access_log_t *access_log);
216630Svbart@nginx.com static u_char *nxt_router_access_log_date(u_char *buf, nxt_realtime_t *now,
217630Svbart@nginx.com     struct tm *tm, size_t size, const char *format);
218630Svbart@nginx.com static void nxt_router_access_log_open(nxt_task_t *task,
219630Svbart@nginx.com     nxt_router_temp_conf_t *tmcf);
220630Svbart@nginx.com static void nxt_router_access_log_ready(nxt_task_t *task,
221630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
222630Svbart@nginx.com static void nxt_router_access_log_error(nxt_task_t *task,
223630Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
224630Svbart@nginx.com static void nxt_router_access_log_release(nxt_task_t *task,
225630Svbart@nginx.com     nxt_thread_spinlock_t *lock, nxt_router_access_log_t *access_log);
226651Svbart@nginx.com static void nxt_router_access_log_reopen_completion(nxt_task_t *task, void *obj,
227651Svbart@nginx.com     void *data);
228631Svbart@nginx.com static void nxt_router_access_log_reopen_ready(nxt_task_t *task,
229631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
230631Svbart@nginx.com static void nxt_router_access_log_reopen_error(nxt_task_t *task,
231631Svbart@nginx.com     nxt_port_recv_msg_t *msg, void *data);
232630Svbart@nginx.com 
233343Smax.romanov@nginx.com static void nxt_router_app_port_ready(nxt_task_t *task,
234343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
235343Smax.romanov@nginx.com static void nxt_router_app_port_error(nxt_task_t *task,
236343Smax.romanov@nginx.com     nxt_port_recv_msg_t *msg, void *data);
237343Smax.romanov@nginx.com 
238507Smax.romanov@nginx.com static void nxt_router_app_quit(nxt_task_t *task, nxt_app_t *app);
239343Smax.romanov@nginx.com static void nxt_router_app_port_release(nxt_task_t *task, nxt_port_t *port,
240343Smax.romanov@nginx.com     uint32_t request_failed, uint32_t got_response);
241427Smax.romanov@nginx.com static nxt_int_t nxt_router_app_port(nxt_task_t *task, nxt_app_t *app,
242427Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
243141Smax.romanov@nginx.com 
244425Smax.romanov@nginx.com static void nxt_router_app_prepare_request(nxt_task_t *task,
245343Smax.romanov@nginx.com     nxt_req_app_link_t *ra);
246216Sigor@sysoev.ru static nxt_int_t nxt_python_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
247216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
248216Sigor@sysoev.ru static nxt_int_t nxt_php_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
249216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
250216Sigor@sysoev.ru static nxt_int_t nxt_go_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
251216Sigor@sysoev.ru     nxt_app_wmsg_t *wmsg);
252510Salexander.borisov@nginx.com static nxt_int_t nxt_perl_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
253510Salexander.borisov@nginx.com     nxt_app_wmsg_t *wmsg);
254584Salexander.borisov@nginx.com static nxt_int_t nxt_ruby_prepare_msg(nxt_task_t *task, nxt_app_request_t *r,
255584Salexander.borisov@nginx.com     nxt_app_wmsg_t *wmsg);
256510Salexander.borisov@nginx.com 
25753Sigor@sysoev.ru static void nxt_router_conn_free(nxt_task_t *task, void *obj, void *data);
258318Smax.romanov@nginx.com static void nxt_router_app_timeout(nxt_task_t *task, void *obj, void *data);
259507Smax.romanov@nginx.com static void nxt_router_adjust_idle_timer(nxt_task_t *task, void *obj,
260507Smax.romanov@nginx.com     void *data);
261507Smax.romanov@nginx.com static void nxt_router_app_idle_timeout(nxt_task_t *task, void *obj,
262507Smax.romanov@nginx.com     void *data);
263507Smax.romanov@nginx.com static void nxt_router_app_release_handler(nxt_task_t *task, void *obj,
264507Smax.romanov@nginx.com     void *data);
265431Sigor@sysoev.ru 
266431Sigor@sysoev.ru static const nxt_http_request_state_t  nxt_http_request_send_state;
267431Sigor@sysoev.ru static void nxt_http_request_send_body(nxt_task_t *task, void *obj, void *data);
268141Smax.romanov@nginx.com 
269119Smax.romanov@nginx.com static nxt_router_t  *nxt_router;
27020Sigor@sysoev.ru 
271216Sigor@sysoev.ru 
272216Sigor@sysoev.ru static nxt_app_prepare_msg_t  nxt_app_prepare_msg[] = {
273216Sigor@sysoev.ru     nxt_python_prepare_msg,
274216Sigor@sysoev.ru     nxt_php_prepare_msg,
275216Sigor@sysoev.ru     nxt_go_prepare_msg,
276510Salexander.borisov@nginx.com     nxt_perl_prepare_msg,
277584Salexander.borisov@nginx.com     nxt_ruby_prepare_msg,
278216Sigor@sysoev.ru };
279216Sigor@sysoev.ru 
280216Sigor@sysoev.ru 
281*662Smax.romanov@nginx.com nxt_port_handlers_t  nxt_router_process_port_handlers = {
282*662Smax.romanov@nginx.com     .quit         = nxt_worker_process_quit_handler,
283*662Smax.romanov@nginx.com     .new_port     = nxt_router_new_port_handler,
284*662Smax.romanov@nginx.com     .change_file  = nxt_port_change_log_file_handler,
285*662Smax.romanov@nginx.com     .mmap         = nxt_port_mmap_handler,
286*662Smax.romanov@nginx.com     .data         = nxt_router_conf_data_handler,
287*662Smax.romanov@nginx.com     .remove_pid   = nxt_router_remove_pid_handler,
288*662Smax.romanov@nginx.com     .access_log   = nxt_router_access_log_reopen_handler,
289*662Smax.romanov@nginx.com     .rpc_ready    = nxt_port_rpc_handler,
290*662Smax.romanov@nginx.com     .rpc_error    = nxt_port_rpc_handler,
291*662Smax.romanov@nginx.com };
292*662Smax.romanov@nginx.com 
293*662Smax.romanov@nginx.com 
29420Sigor@sysoev.ru nxt_int_t
295141Smax.romanov@nginx.com nxt_router_start(nxt_task_t *task, void *data)
29620Sigor@sysoev.ru {
297141Smax.romanov@nginx.com     nxt_int_t      ret;
298*662Smax.romanov@nginx.com     nxt_port_t     *controller_port;
299141Smax.romanov@nginx.com     nxt_router_t   *router;
300141Smax.romanov@nginx.com     nxt_runtime_t  *rt;
301141Smax.romanov@nginx.com 
302141Smax.romanov@nginx.com     rt = task->thread->runtime;
30353Sigor@sysoev.ru 
304431Sigor@sysoev.ru     ret = nxt_http_init(task, rt);
30588Smax.romanov@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
30688Smax.romanov@nginx.com         return ret;
30788Smax.romanov@nginx.com     }
30888Smax.romanov@nginx.com 
30953Sigor@sysoev.ru     router = nxt_zalloc(sizeof(nxt_router_t));
31053Sigor@sysoev.ru     if (nxt_slow_path(router == NULL)) {
31153Sigor@sysoev.ru         return NXT_ERROR;
31253Sigor@sysoev.ru     }
31353Sigor@sysoev.ru 
31453Sigor@sysoev.ru     nxt_queue_init(&router->engines);
31553Sigor@sysoev.ru     nxt_queue_init(&router->sockets);
316133Sigor@sysoev.ru     nxt_queue_init(&router->apps);
31753Sigor@sysoev.ru 
318119Smax.romanov@nginx.com     nxt_router = router;
319119Smax.romanov@nginx.com 
320*662Smax.romanov@nginx.com     controller_port = rt->port_by_type[NXT_PROCESS_CONTROLLER];
321*662Smax.romanov@nginx.com     if (controller_port != NULL) {
322*662Smax.romanov@nginx.com         nxt_router_greet_controller(task, controller_port);
323*662Smax.romanov@nginx.com     }
324*662Smax.romanov@nginx.com 
325115Sigor@sysoev.ru     return NXT_OK;
326115Sigor@sysoev.ru }
327115Sigor@sysoev.ru 
328115Sigor@sysoev.ru 
329343Smax.romanov@nginx.com static void
330*662Smax.romanov@nginx.com nxt_router_greet_controller(nxt_task_t *task, nxt_port_t *controller_port)
331*662Smax.romanov@nginx.com {
332*662Smax.romanov@nginx.com     nxt_port_socket_write(task, controller_port, NXT_PORT_MSG_PROCESS_READY,
333*662Smax.romanov@nginx.com                           -1, 0, 0, NULL);
334*662Smax.romanov@nginx.com }
335*662Smax.romanov@nginx.com 
336*662Smax.romanov@nginx.com 
337*662Smax.romanov@nginx.com static void
338507Smax.romanov@nginx.com nxt_router_start_app_process_handler(nxt_task_t *task, nxt_port_t *port,
339507Smax.romanov@nginx.com     void *data)
340167Smax.romanov@nginx.com {
341343Smax.romanov@nginx.com     size_t         size;
342343Smax.romanov@nginx.com     uint32_t       stream;
343430Sigor@sysoev.ru     nxt_mp_t       *mp;
344648Svbart@nginx.com     nxt_int_t      ret;
345343Smax.romanov@nginx.com     nxt_app_t      *app;
346343Smax.romanov@nginx.com     nxt_buf_t      *b;
347343Smax.romanov@nginx.com     nxt_port_t     *main_port;
348343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
349343Smax.romanov@nginx.com 
350343Smax.romanov@nginx.com     app = data;
351167Smax.romanov@nginx.com 
352167Smax.romanov@nginx.com     rt = task->thread->runtime;
353240Sigor@sysoev.ru     main_port = rt->port_by_type[NXT_PROCESS_MAIN];
354167Smax.romanov@nginx.com 
355507Smax.romanov@nginx.com     nxt_debug(task, "app '%V' %p start process", &app->name, app);
356343Smax.romanov@nginx.com 
357343Smax.romanov@nginx.com     size = app->name.length + 1 + app->conf.length;
358343Smax.romanov@nginx.com 
359343Smax.romanov@nginx.com     b = nxt_buf_mem_ts_alloc(task, task->thread->engine->mem_pool, size);
360343Smax.romanov@nginx.com 
361343Smax.romanov@nginx.com     if (nxt_slow_path(b == NULL)) {
362343Smax.romanov@nginx.com         goto failed;
363167Smax.romanov@nginx.com     }
364167Smax.romanov@nginx.com 
365343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->name);
366343Smax.romanov@nginx.com     *b->mem.free++ = '\0';
367343Smax.romanov@nginx.com     nxt_buf_cpystr(b, &app->conf);
368343Smax.romanov@nginx.com 
369343Smax.romanov@nginx.com     stream = nxt_port_rpc_register_handler(task, port,
370343Smax.romanov@nginx.com                                            nxt_router_app_port_ready,
371343Smax.romanov@nginx.com                                            nxt_router_app_port_error,
372343Smax.romanov@nginx.com                                            -1, app);
373343Smax.romanov@nginx.com 
374343Smax.romanov@nginx.com     if (nxt_slow_path(stream == 0)) {
375343Smax.romanov@nginx.com         goto failed;
376343Smax.romanov@nginx.com     }
377343Smax.romanov@nginx.com 
378648Svbart@nginx.com     ret = nxt_port_socket_write(task, main_port, NXT_PORT_MSG_START_WORKER, -1,
379648Svbart@nginx.com                                 stream, port->id, b);
380648Svbart@nginx.com 
381648Svbart@nginx.com     if (nxt_slow_path(ret != NXT_OK)) {
382648Svbart@nginx.com         nxt_port_rpc_cancel(task, port, stream);
383648Svbart@nginx.com         goto failed;
384648Svbart@nginx.com     }
385343Smax.romanov@nginx.com 
386343Smax.romanov@nginx.com     return;
387343Smax.romanov@nginx.com 
388343Smax.romanov@nginx.com failed:
389343Smax.romanov@nginx.com 
390648Svbart@nginx.com     if (b != NULL) {
391648Svbart@nginx.com         mp = b->data;
392648Svbart@nginx.com         nxt_mp_free(mp, b);
393648Svbart@nginx.com         nxt_mp_release(mp);
394648Svbart@nginx.com     }
395648Svbart@nginx.com 
396343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
397343Smax.romanov@nginx.com 
398507Smax.romanov@nginx.com     app->pending_processes--;
399343Smax.romanov@nginx.com 
400343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
401343Smax.romanov@nginx.com 
402343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
403167Smax.romanov@nginx.com }
404167Smax.romanov@nginx.com 
405167Smax.romanov@nginx.com 
406343Smax.romanov@nginx.com static nxt_int_t
407507Smax.romanov@nginx.com nxt_router_start_app_process(nxt_task_t *task, nxt_app_t *app)
408141Smax.romanov@nginx.com {
409343Smax.romanov@nginx.com     nxt_int_t      res;
410343Smax.romanov@nginx.com     nxt_port_t     *router_port;
411343Smax.romanov@nginx.com     nxt_runtime_t  *rt;
412343Smax.romanov@nginx.com 
413343Smax.romanov@nginx.com     rt = task->thread->runtime;
414343Smax.romanov@nginx.com     router_port = rt->port_by_type[NXT_PROCESS_ROUTER];
415343Smax.romanov@nginx.com 
416343Smax.romanov@nginx.com     nxt_router_app_use(task, app, 1);
417343Smax.romanov@nginx.com 
418507Smax.romanov@nginx.com     res = nxt_port_post(task, router_port, nxt_router_start_app_process_handler,
419343Smax.romanov@nginx.com                         app);
420343Smax.romanov@nginx.com 
421343Smax.romanov@nginx.com     if (res == NXT_OK) {
422343Smax.romanov@nginx.com         return res;
423318Smax.romanov@nginx.com     }
424318Smax.romanov@nginx.com 
425343Smax.romanov@nginx.com     nxt_thread_mutex_lock(&app->mutex);
426343Smax.romanov@nginx.com 
427507Smax.romanov@nginx.com     app->pending_processes--;
428343Smax.romanov@nginx.com 
429343Smax.romanov@nginx.com     nxt_thread_mutex_unlock(&app->mutex);
430343Smax.romanov@nginx.com 
431343Smax.romanov@nginx.com     nxt_router_app_use(task, app, -1);
432343Smax.romanov@nginx.com 
433343Smax.romanov@nginx.com     return NXT_ERROR;
434318Smax.romanov@nginx.com }
435318Smax.romanov@nginx.com 
436318Smax.romanov@nginx.com 
437351Smax.romanov@nginx.com nxt_inline void
438351Smax.romanov@nginx.com nxt_router_ra_init(nxt_task_t *task, nxt_req_app_link_t *ra,
439351Smax.romanov@nginx.com     nxt_req_conn_link_t *rc)
440167Smax.romanov@nginx.com {
441318Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
442351Smax.romanov@nginx.com 
443318Smax.romanov@nginx.com     engine = task->thread->engine;
444167Smax.romanov@nginx.com 
445167Smax.romanov@nginx.com     nxt_memzero(ra, sizeof(nxt_req_app_link_t));
446167Smax.romanov@nginx.com 
447318Smax.romanov@nginx.com     ra->stream = rc->stream;
448425Smax.romanov@nginx.com     ra->use_count = 1;
449167Smax.romanov@nginx.com     ra->rc = rc;
450318Smax.romanov@nginx.com     rc->ra = ra;
451318Smax.romanov@nginx.com     ra->reply_port = engine->port;
452351Smax.romanov@nginx.com     ra->ap = rc->ap;
453167Smax.romanov@nginx.com 
454167Smax.romanov@nginx.com     ra->work.handler = NULL;
455318Smax.romanov@nginx.com     ra->work.task = &engine->task;
456167Smax.romanov@nginx.com     ra->work.obj = ra;
457318Smax.romanov@nginx.com     ra->work.data = engine;
458351Smax.romanov@nginx.com }
459351Smax.romanov@nginx.com 
460351Smax.romanov@nginx.com 
461351Smax.romanov@nginx.com nxt_inline nxt_req_app_link_t *
462351Smax.romanov@nginx.com nxt_router_ra_create(nxt_task_t *task, nxt_req_app_link_t *ra_src)
463351Smax.romanov@nginx.com {
464351Smax.romanov@nginx.com     nxt_mp_t            *mp;
465351Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
466351Smax.romanov@nginx.com 
467425Smax.romanov@nginx.com     if (ra_src->mem_pool != NULL) {
468425Smax.romanov@nginx.com         return ra_src;
469425Smax.romanov@nginx.com     }
470425Smax.romanov@nginx.com 
471351Smax.romanov@nginx.com     mp = ra_src->ap->mem_pool;
472351Smax.romanov@nginx.com 
473430Sigor@sysoev.ru     ra = nxt_mp_alloc(mp, sizeof(nxt_req_app_link_t));
474351Smax.romanov@nginx.com 
475351Smax.romanov@nginx.com     if (nxt_slow_path(ra == NULL)) {
476351Smax.romanov@nginx.com 
477351Smax.romanov@nginx.com         ra_src->rc->ra = NULL;
478351Smax.romanov@nginx.com         ra_src->rc = NULL;
479351Smax.romanov@nginx.com 
480351Smax.romanov@nginx.com         return NULL;
481351Smax.romanov@nginx.com     }
482351Smax.romanov@nginx.com 
483430Sigor@sysoev.ru     nxt_mp_retain(mp);
484430Sigor@sysoev.ru 
485351Smax.romanov@nginx.com     nxt_router_ra_init(task, ra, ra_src->rc);
486351Smax.romanov@nginx.com 
487351Smax.romanov@nginx.com     ra->mem_pool = mp;
488167Smax.romanov@nginx.com 
489167Smax.romanov@nginx.com     return ra;
490167Smax.romanov@nginx.com }
491167Smax.romanov@nginx.com 
492167Smax.romanov@nginx.com 
493423Smax.romanov@nginx.com nxt_inline nxt_bool_t
494423Smax.romanov@nginx.com nxt_router_msg_cancel(nxt_task_t *task, nxt_msg_info_t *msg_info,
495423Smax.romanov@nginx.com     uint32_t stream)
496423Smax.romanov@nginx.com {
497423Smax.romanov@nginx.com     nxt_buf_t   *b, *next;
498423Smax.romanov@nginx.com     nxt_bool_t  cancelled;
499423Smax.romanov@nginx.com 
500423Smax.romanov@nginx.com     if (msg_info->buf == NULL) {
501423Smax.romanov@nginx.com         return 0;
502423Smax.romanov@nginx.com     }
503423Smax.romanov@nginx.com 
504423Smax.romanov@nginx.com     cancelled = nxt_port_mmap_tracking_cancel(task, &msg_info->tracking,
505423Smax.romanov@nginx.com                                               stream);
506423Smax.romanov@nginx.com 
507423Smax.romanov@nginx.com     if (cancelled) {
508423Smax.romanov@nginx.com         nxt_debug(task, "stream #%uD: cancelled by router", stream);
509423Smax.romanov@nginx.com     }
510423Smax.romanov@nginx.com 
511423Smax.romanov@nginx.com     for (b = msg_info->buf; b != NULL; b = next) {
512423Smax.romanov@nginx.com         next = b->next;
513423Smax.romanov@nginx.com 
514423Smax.romanov@nginx.com         b->completion_handler = msg_info->completion_handler;
515423Smax.romanov@nginx.com 
516423Smax.romanov@nginx.com         if (b->is_port_mmap_sent) {
517423Smax.romanov@nginx.com             b->is_port_mmap_sent = cancelled == 0;
518423Smax.romanov@nginx.com             b->completion_handler(task, b, b->parent);
519423Smax.romanov@nginx.com         }
520423Smax.romanov@nginx.com     }
521423Smax.romanov@nginx.com 
522423Smax.romanov@nginx.com     msg_info->buf = NULL;
523423Smax.romanov@nginx.com 
524423Smax.romanov@nginx.com     return cancelled;
525423Smax.romanov@nginx.com }
526423Smax.romanov@nginx.com 
527423Smax.romanov@nginx.com 
528167Smax.romanov@nginx.com static void
529425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra);
530425Smax.romanov@nginx.com 
531425Smax.romanov@nginx.com 
532425Smax.romanov@nginx.com static void
533425Smax.romanov@nginx.com nxt_router_ra_update_peer_handler(nxt_task_t *task, void *obj, void *data)
534167Smax.romanov@nginx.com {
535425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
536425Smax.romanov@nginx.com 
537425Smax.romanov@nginx.com     ra = obj;
538425Smax.romanov@nginx.com 
539425Smax.romanov@nginx.com     nxt_router_ra_update_peer(task, ra);
540425Smax.romanov@nginx.com 
541425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
542425Smax.romanov@nginx.com }
543425Smax.romanov@nginx.com 
544425Smax.romanov@nginx.com 
545425Smax.romanov@nginx.com static void
546425Smax.romanov@nginx.com nxt_router_ra_update_peer(nxt_task_t *task, nxt_req_app_link_t *ra)
547425Smax.romanov@nginx.com {
548343Smax.romanov@nginx.com     nxt_event_engine_t   *engine;
549343Smax.romanov@nginx.com     nxt_req_conn_link_t  *rc;
550318Smax.romanov@nginx.com 
551425Smax.romanov@nginx.com     engine = ra->work.data;
552318Smax.romanov@nginx.com 
553343Smax.romanov@nginx.com     if (task->thread->engine != engine) {
554425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
555425Smax.romanov@nginx.com 
556425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_update_peer_handler;
557318Smax.romanov@nginx.com         ra->work.task = &engine->task;
558318Smax.romanov@nginx.com         ra->work.next = NULL;
559318Smax.romanov@nginx.com 
560425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post update peer to %p",
561318Smax.romanov@nginx.com                   ra->stream, engine);
562318Smax.romanov@nginx.com 
563318Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
564318Smax.romanov@nginx.com 
565318Smax.romanov@nginx.com         return;
566318Smax.romanov@nginx.com     }
567318Smax.romanov@nginx.com 
568425Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD update peer", ra->stream);
569425Smax.romanov@nginx.com 
570425Smax.romanov@nginx.com     rc = ra->rc;
571425Smax.romanov@nginx.com 
572425Smax.romanov@nginx.com     if (rc != NULL && ra->app_port != NULL) {
573425Smax.romanov@nginx.com         nxt_port_rpc_ex_set_peer(task, engine->port, rc, ra->app_port->pid);
574425Smax.romanov@nginx.com     }
575425Smax.romanov@nginx.com 
576425Smax.romanov@nginx.com     nxt_router_ra_use(task, ra, -1);
577425Smax.romanov@nginx.com }
578425Smax.romanov@nginx.com 
579425Smax.romanov@nginx.com 
580425Smax.romanov@nginx.com static void
581425Smax.romanov@nginx.com nxt_router_ra_release(nxt_task_t *task, nxt_req_app_link_t *ra)
582425Smax.romanov@nginx.com {
583431Sigor@sysoev.ru     nxt_mp_t                *mp;
584431Sigor@sysoev.ru     nxt_req_conn_link_t     *rc;
585425Smax.romanov@nginx.com 
586425Smax.romanov@nginx.com     nxt_assert(task->thread->engine == ra->work.data);
587425Smax.romanov@nginx.com     nxt_assert(ra->use_count == 0);
588425Smax.romanov@nginx.com 
589343Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD release", ra->stream);
590343Smax.romanov@nginx.com 
591343Smax.romanov@nginx.com     rc = ra->rc;
592343Smax.romanov@nginx.com 
593343Smax.romanov@nginx.com     if (rc != NULL) {
594423Smax.romanov@nginx.com         if (nxt_slow_path(ra->err_code != 0)) {
595431Sigor@sysoev.ru             nxt_http_request_error(task, rc->ap->request, ra->err_code);
596423Smax.romanov@nginx.com 
597423Smax.romanov@nginx.com         } else {
598423Smax.romanov@nginx.com             rc->app_port = ra->app_port;
599423Smax.romanov@nginx.com             rc->msg_info = ra->msg_info;
600423Smax.romanov@nginx.com 
601425Smax.romanov@nginx.com             if (rc->app->timeout != 0) {
602431Sigor@sysoev.ru                 rc->ap->timer.handler = nxt_router_app_timeout;
603615Smax.romanov@nginx.com                 rc->ap->timer_data = rc;
604431Sigor@sysoev.ru                 nxt_timer_add(task->thread->engine, &rc->ap->timer,
605425Smax.romanov@nginx.com                               rc->app->timeout);
606425Smax.romanov@nginx.com             }
607425Smax.romanov@nginx.com 
608423Smax.romanov@nginx.com             ra->app_port = NULL;
609423Smax.romanov@nginx.com             ra->msg_info.buf = NULL;
610423Smax.romanov@nginx.com         }
611343Smax.romanov@nginx.com 
612343Smax.romanov@nginx.com         rc->ra = NULL;
613343Smax.romanov@nginx.com         ra->rc = NULL;
614343Smax.romanov@nginx.com     }
615343Smax.romanov@nginx.com 
616343Smax.romanov@nginx.com     if (ra->app_port != NULL) {
617343Smax.romanov@nginx.com         nxt_router_app_port_release(task, ra->app_port, 0, 1);
618343Smax.romanov@nginx.com 
619343Smax.romanov@nginx.com         ra->app_port = NULL;
620167Smax.romanov@nginx.com     }
621167Smax.romanov@nginx.com 
622423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &ra->msg_info, ra->stream);
623423Smax.romanov@nginx.com 
624430Sigor@sysoev.ru     mp = ra->mem_pool;
625430Sigor@sysoev.ru 
626430Sigor@sysoev.ru     if (mp != NULL) {
627430Sigor@sysoev.ru         nxt_mp_free(mp, ra);
628430Sigor@sysoev.ru         nxt_mp_release(mp);
629351Smax.romanov@nginx.com     }
630167Smax.romanov@nginx.com }
631167Smax.romanov@nginx.com 
632167Smax.romanov@nginx.com 
633425Smax.romanov@nginx.com static void
634425Smax.romanov@nginx.com nxt_router_ra_release_handler(nxt_task_t *task, void *obj, void *data)
635425Smax.romanov@nginx.com {
636425Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
637425Smax.romanov@nginx.com 
638425Smax.romanov@nginx.com     ra = obj;
639425Smax.romanov@nginx.com 
640425Smax.romanov@nginx.com     nxt_assert(ra->work.data == data);
641425Smax.romanov@nginx.com 
642425Smax.romanov@nginx.com     nxt_atomic_fetch_add(&ra->use_count, -1);
643425Smax.romanov@nginx.com 
644425Smax.romanov@nginx.com     nxt_router_ra_release(task, ra);
645425Smax.romanov@nginx.com }
646425Smax.romanov@nginx.com 
647425Smax.romanov@nginx.com 
648425Smax.romanov@nginx.com static void
649425Smax.romanov@nginx.com nxt_router_ra_use(nxt_task_t *task, nxt_req_app_link_t *ra, int i)
650425Smax.romanov@nginx.com {
651425Smax.romanov@nginx.com     int                 c;
652425Smax.romanov@nginx.com     nxt_event_engine_t  *engine;
653425Smax.romanov@nginx.com 
654425Smax.romanov@nginx.com     c = nxt_atomic_fetch_add(&ra->use_count, i);
655425Smax.romanov@nginx.com 
656425Smax.romanov@nginx.com     if (i < 0 && c == -i) {
657425Smax.romanov@nginx.com         engine = ra->work.data;
658425Smax.romanov@nginx.com 
659425Smax.romanov@nginx.com         if (task->thread->engine == engine) {
660425Smax.romanov@nginx.com             nxt_router_ra_release(task, ra);
661425Smax.romanov@nginx.com 
662425Smax.romanov@nginx.com             return;
663425Smax.romanov@nginx.com         }
664425Smax.romanov@nginx.com 
665425Smax.romanov@nginx.com         nxt_router_ra_inc_use(ra);
666425Smax.romanov@nginx.com 
667425Smax.romanov@nginx.com         ra->work.handler = nxt_router_ra_release_handler;
668425Smax.romanov@nginx.com         ra->work.task = &engine->task;
669425Smax.romanov@nginx.com         ra->work.next = NULL;
670425Smax.romanov@nginx.com 
671425Smax.romanov@nginx.com         nxt_debug(task, "ra stream #%uD post release to %p",
672425Smax.romanov@nginx.com                   ra->stream, engine);
673425Smax.romanov@nginx.com 
674425Smax.romanov@nginx.com         nxt_event_engine_post(engine, &ra->work);
675425Smax.romanov@nginx.com     }
676425Smax.romanov@nginx.com }
677425Smax.romanov@nginx.com 
678425Smax.romanov@nginx.com 
679423Smax.romanov@nginx.com nxt_inline void
680521Szelenkov@nginx.com nxt_router_ra_error(nxt_req_app_link_t *ra, int code, const char *str)
681345Smax.romanov@nginx.com {
682423Smax.romanov@nginx.com     ra->app_port = NULL;
683423Smax.romanov@nginx.com     ra->err_code = code;
684423Smax.romanov@nginx.com     ra->err_str = str;
685345Smax.romanov@nginx.com }
686345Smax.romanov@nginx.com 
687345Smax.romanov@nginx.com 
688427Smax.romanov@nginx.com nxt_inline void
689427Smax.romanov@nginx.com nxt_router_ra_pending(nxt_task_t *task, nxt_app_t *app, nxt_req_app_link_t *ra)
690427Smax.romanov@nginx.com {
691427Smax.romanov@nginx.com     nxt_queue_insert_tail(&ra->app_port->pending_requests,
692427Smax.romanov@nginx.com                           &ra->link_port_pending);
693427Smax.romanov@nginx.com     nxt_queue_insert_tail(&app->pending, &ra->link_app_pending);
694427Smax.romanov@nginx.com 
695427Smax.romanov@nginx.com     nxt_router_ra_inc_use(ra);
696427Smax.romanov@nginx.com 
697427Smax.romanov@nginx.com     ra->res_time = nxt_thread_monotonic_time(task->thread) + app->res_timeout;
698427Smax.romanov@nginx.com 
699427Smax.romanov@nginx.com     nxt_debug(task, "ra stream #%uD enqueue to pending_requests", ra->stream);
700427Smax.romanov@nginx.com }
701427Smax.romanov@nginx.com 
702427Smax.romanov@nginx.com 
703425Smax.romanov@nginx.com nxt_inline nxt_bool_t
704425Smax.romanov@nginx.com nxt_queue_chk_remove(nxt_queue_link_t *lnk)
705425Smax.romanov@nginx.com {
706425Smax.romanov@nginx.com     if (lnk->next != NULL) {
707425Smax.romanov@nginx.com         nxt_queue_remove(lnk);
708425Smax.romanov@nginx.com 
709425Smax.romanov@nginx.com         lnk->next = NULL;
710425Smax.romanov@nginx.com 
711425Smax.romanov@nginx.com         return 1;
712425Smax.romanov@nginx.com     }
713425Smax.romanov@nginx.com 
714425Smax.romanov@nginx.com     return 0;
715425Smax.romanov@nginx.com }
716425Smax.romanov@nginx.com 
717425Smax.romanov@nginx.com 
718343Smax.romanov@nginx.com nxt_inline void
719343Smax.romanov@nginx.com nxt_router_rc_unlink(nxt_task_t *task, nxt_req_conn_link_t *rc)
720343Smax.romanov@nginx.com {
721425Smax.romanov@nginx.com     int                 ra_use_delta;
722343Smax.romanov@nginx.com     nxt_req_app_link_t  *ra;
723343Smax.romanov@nginx.com 
724343Smax.romanov@nginx.com     if (rc->app_port != NULL) {
725343Smax.romanov@nginx.com         nxt_router_app_port_release(task, rc->app_port, 0, 1);
726343Smax.romanov@nginx.com 
727343Smax.romanov@nginx.com         rc->app_port = NULL;
728343Smax.romanov@nginx.com     }
729343Smax.romanov@nginx.com 
730423Smax.romanov@nginx.com     nxt_router_msg_cancel(task, &rc->msg_info, rc->stream);
731423Smax.romanov@nginx.com 
732343Smax.romanov@nginx.com     ra = rc->ra;
733343Smax.romanov@nginx.com 
734343Smax.romanov@nginx.com     if (ra != NULL) {
735343Smax.romanov@nginx.com         rc->ra = NULL;
736343Smax.romanov@nginx.com         ra->rc = NULL;
737343Smax.romanov@nginx.com 
738425Smax.romanov@nginx.com         ra_use_delta = 0;
739425Smax.romanov@nginx.com 
740343Smax.romanov@nginx.com         nxt_thread_mutex_lock(&rc->app->mutex);
741343Smax.romanov@nginx.com 
742425Smax.romanov@nginx.com         if (ra->link_app_requests.next == NULL
743427Smax.romanov@nginx.com             && ra->link_port_pending.next == NULL
744427Smax.romanov@nginx.com             && ra->link_app_pending.next == NULL)
745425Smax.romanov@nginx.com         {
746425Smax.romanov@nginx.com             ra = NULL;
747343Smax.romanov@nginx.com 
748343Smax.romanov@nginx.com         } else {
749425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_app_requests);
750425Smax.romanov@nginx.com             ra_use_delta -= nxt_queue_chk_remove(&ra->link_port_pending);
751427Smax.romanov@nginx.com             nxt_queue_chk_remove(&ra->link_app_pending);
752343Smax.romanov@nginx.com         }
753343Smax.romanov@nginx.com 
754343Smax.romanov@nginx.com         nxt_thread_mutex_unlock(&rc->app->mutex);
755425Smax.romanov@nginx.com 
756425Smax.romanov@nginx.com         if (ra != NULL) {
757425Smax.romanov@nginx.com             nxt_router_ra_use(task, ra, ra_use_delta);
758425Smax.romanov@nginx.com         }
759343Smax.romanov@nginx.com     }
760343Smax.romanov@nginx.com 
761343Smax.romanov@nginx.com     if (rc->app != NULL) {
762343Smax.romanov@nginx.com         nxt_router_app_use(task, rc->app, -1);
763343Smax.romanov@nginx.com 
764343Smax.romanov@nginx.com         rc->app = NULL;
765343Smax.romanov@nginx.com     }
766343Smax.romanov@nginx.com 
767346Smax.romanov@nginx.com     if (rc->ap != NULL) {
768615Smax.romanov@nginx.com         rc->ap->timer_data = NULL;
769615Smax.romanov@nginx.com 
770346Smax.romanov@nginx.com         nxt_app_http_req_done(task, rc->ap);
771346Smax.romanov@nginx.com 
772346Smax.romanov@nginx.com         rc->ap = NULL;
773346Smax.romanov@nginx.com     }
774343Smax.romanov@nginx.com }
775343Smax.romanov@nginx.com 
776343Smax.romanov@nginx.com 
777141Smax.romanov@nginx.com void
778141Smax.romanov@nginx.com nxt_router_new_port_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg)
779141Smax.romanov@nginx.com {
780141Smax.romanov@nginx.com     nxt_port_new_port_handler(task, msg);
781141Smax.romanov@nginx.com 
782*662Smax.romanov@nginx.com     if (msg->u.new_port->type == NXT_PROCESS_CONTROLLER) {
783*662Smax.romanov@nginx.com         nxt_router_greet_controller(task, msg->u.new_port);
784*662Smax.romanov@nginx.com     }
785*662Smax.romanov@nginx.com 
786192Smax.romanov@nginx.com     if (msg->port_msg.stream == 0) {
787141Smax.romanov@nginx.com         return;
788141Smax.romanov@nginx.com     }
789141Smax.romanov@nginx.com 
790426Smax.romanov@nginx.com     if (msg->u.new_port == NULL
791426Smax.romanov@nginx.com         || msg->u.new_port->type != NXT_PROCESS_WORKER)
792347Smax.romanov@nginx.com     {
793192Smax.romanov@nginx.com         msg->port_msg.type = _NXT_PORT_MSG_RPC_ERROR;
794141Smax.romanov@nginx.com     }
795192