blob: 525ef41d79a0ea2037f5dd9df1a1debafc665213 [file] [log] [blame]
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001
Igor Sysoevd90282d2004-09-28 08:34:51 +00002/*
Igor Sysoevff8da912004-09-29 16:00:49 +00003 * Copyright (C) Igor Sysoev
Igor Sysoevd90282d2004-09-28 08:34:51 +00004 */
5
6
Igor Sysoev3c3ca172004-01-05 20:55:48 +00007#include <ngx_config.h>
8#include <ngx_core.h>
9#include <ngx_event.h>
Igor Sysoev3c3ca172004-01-05 20:55:48 +000010
11
Igor Sysoev9e580192006-02-01 18:22:15 +000012static void ngx_destroy_cycle_pools(ngx_conf_t *conf);
Igor Sysoev02025fd2005-01-18 13:03:58 +000013static ngx_int_t ngx_cmp_sockaddr(struct sockaddr *sa1, struct sockaddr *sa2);
Igor Sysoevf7a08d52009-04-18 19:27:28 +000014static ngx_int_t ngx_init_zone_pool(ngx_cycle_t *cycle,
15 ngx_shm_zone_t *shm_zone);
16static ngx_int_t ngx_test_lockfile(u_char *file, ngx_log_t *log);
Igor Sysoev3c3ca172004-01-05 20:55:48 +000017static void ngx_clean_old_cycles(ngx_event_t *ev);
18
19
20volatile ngx_cycle_t *ngx_cycle;
21ngx_array_t ngx_old_cycles;
22
23static ngx_pool_t *ngx_temp_pool;
24static ngx_event_t ngx_cleaner_event;
25
Igor Sysoev630ad0c2004-04-16 05:14:16 +000026ngx_uint_t ngx_test_config;
Igor Sysoev2cb30f12010-09-02 13:43:02 +000027ngx_uint_t ngx_quiet_mode;
Igor Sysoev630ad0c2004-04-16 05:14:16 +000028
Igor Sysoev2b979932004-07-07 15:01:00 +000029#if (NGX_THREADS)
30ngx_tls_key_t ngx_core_tls_key;
31#endif
32
Igor Sysoev3c3ca172004-01-05 20:55:48 +000033
34/* STUB NAME */
35static ngx_connection_t dumb;
36/* STUB */
37
Igor Sysoev980a9242004-09-05 19:54:02 +000038static ngx_str_t error_log = ngx_string(NGX_ERROR_LOG_PATH);
39
Igor Sysoev3c3ca172004-01-05 20:55:48 +000040
Igor Sysoevc2068d02005-10-19 12:33:58 +000041ngx_cycle_t *
42ngx_init_cycle(ngx_cycle_t *old_cycle)
Igor Sysoev3c3ca172004-01-05 20:55:48 +000043{
Igor Sysoev7f35ae62007-12-16 11:58:16 +000044 void *rv;
45 char **senv, **env;
Igor Sysoev7f35ae62007-12-16 11:58:16 +000046 ngx_uint_t i, n;
47 ngx_log_t *log;
Igor Sysoevfcb5a702008-08-30 19:52:07 +000048 ngx_time_t *tp;
Igor Sysoev7f35ae62007-12-16 11:58:16 +000049 ngx_conf_t conf;
50 ngx_pool_t *pool;
51 ngx_cycle_t *cycle, **old;
52 ngx_shm_zone_t *shm_zone, *oshm_zone;
Igor Sysoev7f35ae62007-12-16 11:58:16 +000053 ngx_list_part_t *part, *opart;
54 ngx_open_file_t *file;
55 ngx_listening_t *ls, *nls;
56 ngx_core_conf_t *ccf, *old_ccf;
57 ngx_core_module_t *module;
Igor Sysoev3be52572008-05-16 14:39:06 +000058 char hostname[NGX_MAXHOSTNAMELEN];
Igor Sysoev3c3ca172004-01-05 20:55:48 +000059
Igor Sysoevfcb5a702008-08-30 19:52:07 +000060 ngx_timezone_update();
61
62 /* force localtime update with a new timezone */
63
64 tp = ngx_timeofday();
65 tp->sec = 0;
66
Igor Sysoev6d45d8a2010-03-25 09:10:10 +000067 ngx_time_update();
Igor Sysoevfcb5a702008-08-30 19:52:07 +000068
69
Igor Sysoev3c3ca172004-01-05 20:55:48 +000070 log = old_cycle->log;
71
Igor Sysoev02f742b2005-04-08 15:18:55 +000072 pool = ngx_create_pool(NGX_CYCLE_POOL_SIZE, log);
Igor Sysoevc1571722005-03-19 12:38:37 +000073 if (pool == NULL) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +000074 return NULL;
75 }
Igor Sysoev68df19d2004-04-15 15:34:36 +000076 pool->log = log;
Igor Sysoev3c3ca172004-01-05 20:55:48 +000077
Igor Sysoevc1571722005-03-19 12:38:37 +000078 cycle = ngx_pcalloc(pool, sizeof(ngx_cycle_t));
79 if (cycle == NULL) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +000080 ngx_destroy_pool(pool);
81 return NULL;
82 }
Igor Sysoevc1571722005-03-19 12:38:37 +000083
Igor Sysoev3c3ca172004-01-05 20:55:48 +000084 cycle->pool = pool;
Igor Sysoevcccc5522004-04-14 20:34:05 +000085 cycle->log = log;
Igor Sysoev0cd76ea2009-04-30 13:53:42 +000086 cycle->new_log.log_level = NGX_LOG_ERR;
Igor Sysoev3c3ca172004-01-05 20:55:48 +000087 cycle->old_cycle = old_cycle;
Igor Sysoevab517d52004-05-18 15:29:08 +000088
Igor Sysoev5ef370d2009-04-27 11:32:33 +000089 cycle->conf_prefix.len = old_cycle->conf_prefix.len;
90 cycle->conf_prefix.data = ngx_pstrdup(pool, &old_cycle->conf_prefix);
91 if (cycle->conf_prefix.data == NULL) {
92 ngx_destroy_pool(pool);
93 return NULL;
94 }
95
96 cycle->prefix.len = old_cycle->prefix.len;
97 cycle->prefix.data = ngx_pstrdup(pool, &old_cycle->prefix);
98 if (cycle->prefix.data == NULL) {
99 ngx_destroy_pool(pool);
100 return NULL;
101 }
Igor Sysoevab517d52004-05-18 15:29:08 +0000102
Igor Sysoev9e580192006-02-01 18:22:15 +0000103 cycle->conf_file.len = old_cycle->conf_file.len;
Igor Sysoev7f6b2ff2008-06-17 15:00:30 +0000104 cycle->conf_file.data = ngx_pnalloc(pool, old_cycle->conf_file.len + 1);
Igor Sysoev9e580192006-02-01 18:22:15 +0000105 if (cycle->conf_file.data == NULL) {
106 ngx_destroy_pool(pool);
107 return NULL;
108 }
109 ngx_cpystrn(cycle->conf_file.data, old_cycle->conf_file.data,
110 old_cycle->conf_file.len + 1);
111
Igor Sysoevb4fbdcf2008-06-30 12:35:16 +0000112 cycle->conf_param.len = old_cycle->conf_param.len;
Igor Sysoev5ef370d2009-04-27 11:32:33 +0000113 cycle->conf_param.data = ngx_pstrdup(pool, &old_cycle->conf_param);
Igor Sysoevb4fbdcf2008-06-30 12:35:16 +0000114 if (cycle->conf_param.data == NULL) {
115 ngx_destroy_pool(pool);
116 return NULL;
117 }
Igor Sysoevb4fbdcf2008-06-30 12:35:16 +0000118
119
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000120 n = old_cycle->pathes.nelts ? old_cycle->pathes.nelts : 10;
Igor Sysoevc1571722005-03-19 12:38:37 +0000121
122 cycle->pathes.elts = ngx_pcalloc(pool, n * sizeof(ngx_path_t *));
123 if (cycle->pathes.elts == NULL) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000124 ngx_destroy_pool(pool);
125 return NULL;
126 }
Igor Sysoevc1571722005-03-19 12:38:37 +0000127
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000128 cycle->pathes.nelts = 0;
129 cycle->pathes.size = sizeof(ngx_path_t *);
130 cycle->pathes.nalloc = n;
131 cycle->pathes.pool = pool;
132
133
Igor Sysoevb9e34412004-09-03 15:50:30 +0000134 if (old_cycle->open_files.part.nelts) {
135 n = old_cycle->open_files.part.nelts;
136 for (part = old_cycle->open_files.part.next; part; part = part->next) {
137 n += part->nelts;
138 }
139
140 } else {
141 n = 20;
142 }
143
Igor Sysoevaab4d8c2004-09-06 18:45:00 +0000144 if (ngx_list_init(&cycle->open_files, pool, n, sizeof(ngx_open_file_t))
Igor Sysoev1f4220e2009-02-24 10:42:23 +0000145 != NGX_OK)
Igor Sysoev980a9242004-09-05 19:54:02 +0000146 {
Igor Sysoevb9e34412004-09-03 15:50:30 +0000147 ngx_destroy_pool(pool);
148 return NULL;
149 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000150
151
Igor Sysoev67cd3362006-11-20 08:51:45 +0000152 if (old_cycle->shared_memory.part.nelts) {
153 n = old_cycle->shared_memory.part.nelts;
154 for (part = old_cycle->shared_memory.part.next; part; part = part->next)
155 {
156 n += part->nelts;
157 }
158
159 } else {
160 n = 1;
161 }
162
163 if (ngx_list_init(&cycle->shared_memory, pool, n, sizeof(ngx_shm_zone_t))
Igor Sysoev1f4220e2009-02-24 10:42:23 +0000164 != NGX_OK)
Igor Sysoev67cd3362006-11-20 08:51:45 +0000165 {
166 ngx_destroy_pool(pool);
167 return NULL;
168 }
169
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000170 n = old_cycle->listening.nelts ? old_cycle->listening.nelts : 10;
Igor Sysoevc1571722005-03-19 12:38:37 +0000171
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000172 cycle->listening.elts = ngx_pcalloc(pool, n * sizeof(ngx_listening_t));
173 if (cycle->listening.elts == NULL) {
174 ngx_destroy_pool(pool);
175 return NULL;
176 }
Igor Sysoevc1571722005-03-19 12:38:37 +0000177
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000178 cycle->listening.nelts = 0;
179 cycle->listening.size = sizeof(ngx_listening_t);
180 cycle->listening.nalloc = n;
181 cycle->listening.pool = pool;
182
183
Igor Sysoev4956ac52011-04-04 12:26:53 +0000184 ngx_queue_init(&cycle->reusable_connections_queue);
185
186
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000187 cycle->conf_ctx = ngx_pcalloc(pool, ngx_max_module * sizeof(void *));
188 if (cycle->conf_ctx == NULL) {
189 ngx_destroy_pool(pool);
190 return NULL;
191 }
192
193
Igor Sysoev3be52572008-05-16 14:39:06 +0000194 if (gethostname(hostname, NGX_MAXHOSTNAMELEN) == -1) {
195 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, "gethostname() failed");
196 ngx_destroy_pool(pool);
197 return NULL;
198 }
199
200 /* on Linux gethostname() silently truncates name that does not fit */
201
202 hostname[NGX_MAXHOSTNAMELEN - 1] = '\0';
203 cycle->hostname.len = ngx_strlen(hostname);
204
Igor Sysoev7f6b2ff2008-06-17 15:00:30 +0000205 cycle->hostname.data = ngx_pnalloc(pool, cycle->hostname.len);
Igor Sysoev3be52572008-05-16 14:39:06 +0000206 if (cycle->hostname.data == NULL) {
207 ngx_destroy_pool(pool);
208 return NULL;
209 }
210
Igor Sysoev42b72992009-10-21 17:04:13 +0000211 ngx_strlow(cycle->hostname.data, (u_char *) hostname, cycle->hostname.len);
Igor Sysoev3be52572008-05-16 14:39:06 +0000212
213
Igor Sysoev43f13192004-04-12 16:38:09 +0000214 for (i = 0; ngx_modules[i]; i++) {
215 if (ngx_modules[i]->type != NGX_CORE_MODULE) {
216 continue;
217 }
218
219 module = ngx_modules[i]->ctx;
220
221 if (module->create_conf) {
222 rv = module->create_conf(cycle);
Igor Sysoev260c4322009-06-02 16:09:44 +0000223 if (rv == NULL) {
Igor Sysoev43f13192004-04-12 16:38:09 +0000224 ngx_destroy_pool(pool);
225 return NULL;
226 }
227 cycle->conf_ctx[ngx_modules[i]->index] = rv;
228 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000229 }
230
231
Igor Sysoev7f35ae62007-12-16 11:58:16 +0000232 senv = environ;
233
234
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000235 ngx_memzero(&conf, sizeof(ngx_conf_t));
236 /* STUB: init array ? */
Igor Sysoevc1571722005-03-19 12:38:37 +0000237 conf.args = ngx_array_create(pool, 10, sizeof(ngx_str_t));
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000238 if (conf.args == NULL) {
239 ngx_destroy_pool(pool);
240 return NULL;
241 }
242
Igor Sysoev305a9d82005-12-26 17:07:48 +0000243 conf.temp_pool = ngx_create_pool(NGX_CYCLE_POOL_SIZE, log);
244 if (conf.temp_pool == NULL) {
245 ngx_destroy_pool(pool);
246 return NULL;
247 }
248
Igor Sysoev9e580192006-02-01 18:22:15 +0000249
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000250 conf.ctx = cycle->conf_ctx;
251 conf.cycle = cycle;
Igor Sysoev68df19d2004-04-15 15:34:36 +0000252 conf.pool = pool;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000253 conf.log = log;
254 conf.module_type = NGX_CORE_MODULE;
255 conf.cmd_type = NGX_MAIN_CONF;
256
Igor Sysoev50cc0c92006-10-02 08:50:03 +0000257#if 0
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000258 log->log_level = NGX_LOG_DEBUG_ALL;
259#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000260
Igor Sysoevb4fbdcf2008-06-30 12:35:16 +0000261 if (ngx_conf_param(&conf) != NGX_CONF_OK) {
Igor Sysoev15c149e2009-09-18 09:21:14 +0000262 environ = senv;
Igor Sysoevb4fbdcf2008-06-30 12:35:16 +0000263 ngx_destroy_cycle_pools(&conf);
264 return NULL;
265 }
266
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000267 if (ngx_conf_parse(&conf, &cycle->conf_file) != NGX_CONF_OK) {
Igor Sysoev15c149e2009-09-18 09:21:14 +0000268 environ = senv;
Igor Sysoev9e580192006-02-01 18:22:15 +0000269 ngx_destroy_cycle_pools(&conf);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000270 return NULL;
271 }
272
Igor Sysoev2cb30f12010-09-02 13:43:02 +0000273 if (ngx_test_config && !ngx_quiet_mode) {
Igor Sysoevc28ff712009-04-23 11:13:12 +0000274 ngx_log_stderr(0, "the configuration file %s syntax is ok",
Igor Sysoev1153aa62009-04-19 16:06:09 +0000275 cycle->conf_file.data);
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000276 }
277
Igor Sysoev43f13192004-04-12 16:38:09 +0000278 for (i = 0; ngx_modules[i]; i++) {
279 if (ngx_modules[i]->type != NGX_CORE_MODULE) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000280 continue;
281 }
282
Igor Sysoev43f13192004-04-12 16:38:09 +0000283 module = ngx_modules[i]->ctx;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000284
Igor Sysoev43f13192004-04-12 16:38:09 +0000285 if (module->init_conf) {
286 if (module->init_conf(cycle, cycle->conf_ctx[ngx_modules[i]->index])
Igor Sysoev305a9d82005-12-26 17:07:48 +0000287 == NGX_CONF_ERROR)
Igor Sysoev43f13192004-04-12 16:38:09 +0000288 {
Igor Sysoev15c149e2009-09-18 09:21:14 +0000289 environ = senv;
Igor Sysoev9e580192006-02-01 18:22:15 +0000290 ngx_destroy_cycle_pools(&conf);
Igor Sysoev43f13192004-04-12 16:38:09 +0000291 return NULL;
292 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000293 }
Igor Sysoev43f13192004-04-12 16:38:09 +0000294 }
295
Igor Sysoev165aa392009-06-06 12:41:31 +0000296 if (ngx_process == NGX_PROCESS_SIGNALLER) {
297 return cycle;
298 }
Igor Sysoev43f13192004-04-12 16:38:09 +0000299
Igor Sysoevffe71442006-02-08 15:33:12 +0000300 ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
Igor Sysoev43f13192004-04-12 16:38:09 +0000301
Igor Sysoevffe71442006-02-08 15:33:12 +0000302 if (ngx_test_config) {
Igor Sysoev43f13192004-04-12 16:38:09 +0000303
Igor Sysoevffe71442006-02-08 15:33:12 +0000304 if (ngx_create_pidfile(&ccf->pid, log) != NGX_OK) {
305 goto failed;
306 }
Igor Sysoevd43bee82004-11-20 19:52:20 +0000307
Igor Sysoevffe71442006-02-08 15:33:12 +0000308 } else if (!ngx_is_init_cycle(old_cycle)) {
309
310 /*
311 * we do not create the pid file in the first ngx_init_cycle() call
312 * because we need to write the demonized process pid
313 */
314
315 old_ccf = (ngx_core_conf_t *) ngx_get_conf(old_cycle->conf_ctx,
316 ngx_core_module);
317 if (ccf->pid.len != old_ccf->pid.len
318 || ngx_strcmp(ccf->pid.data, old_ccf->pid.data) != 0)
319 {
320 /* new pid file name */
321
322 if (ngx_create_pidfile(&ccf->pid, log) != NGX_OK) {
323 goto failed;
324 }
325
326 ngx_delete_pidfile(old_cycle);
Igor Sysoevd43bee82004-11-20 19:52:20 +0000327 }
328 }
329
Igor Sysoevb9e34412004-09-03 15:50:30 +0000330
Igor Sysoev67cd3362006-11-20 08:51:45 +0000331 if (ngx_test_lockfile(cycle->lock_file.data, log) != NGX_OK) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000332 goto failed;
333 }
Igor Sysoev1b735832004-11-11 14:07:14 +0000334
Igor Sysoevb9e34412004-09-03 15:50:30 +0000335
Igor Sysoevffe71442006-02-08 15:33:12 +0000336 if (ngx_create_pathes(cycle, ccf->user) != NGX_OK) {
337 goto failed;
338 }
Igor Sysoevb9e34412004-09-03 15:50:30 +0000339
Igor Sysoevb9e34412004-09-03 15:50:30 +0000340
Igor Sysoev0cd76ea2009-04-30 13:53:42 +0000341 if (cycle->new_log.file == NULL) {
342 cycle->new_log.file = ngx_conf_open_file(cycle, &error_log);
343 if (cycle->new_log.file == NULL) {
344 goto failed;
345 }
346 }
347
Igor Sysoevffe71442006-02-08 15:33:12 +0000348 /* open the new files */
Igor Sysoev43f13192004-04-12 16:38:09 +0000349
Igor Sysoevffe71442006-02-08 15:33:12 +0000350 part = &cycle->open_files.part;
351 file = part->elts;
Igor Sysoev43f13192004-04-12 16:38:09 +0000352
Igor Sysoevffe71442006-02-08 15:33:12 +0000353 for (i = 0; /* void */ ; i++) {
Igor Sysoevb9e34412004-09-03 15:50:30 +0000354
Igor Sysoevffe71442006-02-08 15:33:12 +0000355 if (i >= part->nelts) {
356 if (part->next == NULL) {
Igor Sysoev43f13192004-04-12 16:38:09 +0000357 break;
358 }
Igor Sysoevffe71442006-02-08 15:33:12 +0000359 part = part->next;
360 file = part->elts;
361 i = 0;
362 }
363
Igor Sysoev5ef370d2009-04-27 11:32:33 +0000364 if (file[i].name.len == 0) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000365 continue;
366 }
367
Igor Sysoevfadc7a72009-03-30 14:51:51 +0000368 file[i].fd = ngx_open_file(file[i].name.data,
Igor Sysoev24c27872009-03-31 13:52:01 +0000369 NGX_FILE_APPEND,
Igor Sysoevfadc7a72009-03-30 14:51:51 +0000370 NGX_FILE_CREATE_OR_OPEN,
Igor Sysoev50034b82007-01-18 20:15:09 +0000371 NGX_FILE_DEFAULT_ACCESS);
Igor Sysoevffe71442006-02-08 15:33:12 +0000372
373 ngx_log_debug3(NGX_LOG_DEBUG_CORE, log, 0,
374 "log: %p %d \"%s\"",
375 &file[i], file[i].fd, file[i].name.data);
376
377 if (file[i].fd == NGX_INVALID_FILE) {
378 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
379 ngx_open_file_n " \"%s\" failed",
380 file[i].name.data);
381 goto failed;
382 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000383
Igor Sysoeva2de7b92009-04-08 19:13:28 +0000384#if !(NGX_WIN32)
Igor Sysoevffe71442006-02-08 15:33:12 +0000385 if (fcntl(file[i].fd, F_SETFD, FD_CLOEXEC) == -1) {
386 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
387 "fcntl(FD_CLOEXEC) \"%s\" failed",
388 file[i].name.data);
389 goto failed;
390 }
391#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000392 }
393
Igor Sysoev0cd76ea2009-04-30 13:53:42 +0000394 cycle->log = &cycle->new_log;
395 pool->log = &cycle->new_log;
Igor Sysoev9aa30bc2009-01-26 14:31:49 +0000396
Igor Sysoev1b735832004-11-11 14:07:14 +0000397
Igor Sysoev67cd3362006-11-20 08:51:45 +0000398 /* create shared memory */
399
400 part = &cycle->shared_memory.part;
Igor Sysoev28c7f762007-01-02 23:54:14 +0000401 shm_zone = part->elts;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000402
403 for (i = 0; /* void */ ; i++) {
404
405 if (i >= part->nelts) {
406 if (part->next == NULL) {
407 break;
408 }
409 part = part->next;
Igor Sysoev28c7f762007-01-02 23:54:14 +0000410 shm_zone = part->elts;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000411 i = 0;
412 }
413
Igor Sysoev14110b32007-01-06 18:52:06 +0000414 if (shm_zone[i].shm.size == 0) {
415 ngx_log_error(NGX_LOG_EMERG, log, 0,
416 "zero size shared memory zone \"%V\"",
Igor Sysoevc7f876b2009-04-16 19:25:09 +0000417 &shm_zone[i].shm.name);
Igor Sysoev14110b32007-01-06 18:52:06 +0000418 goto failed;
419 }
420
Igor Sysoev28c7f762007-01-02 23:54:14 +0000421 shm_zone[i].shm.log = cycle->log;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000422
423 opart = &old_cycle->shared_memory.part;
Igor Sysoev28c7f762007-01-02 23:54:14 +0000424 oshm_zone = opart->elts;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000425
426 for (n = 0; /* void */ ; n++) {
427
428 if (n >= opart->nelts) {
429 if (opart->next == NULL) {
430 break;
431 }
432 opart = opart->next;
Igor Sysoev28c7f762007-01-02 23:54:14 +0000433 oshm_zone = opart->elts;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000434 n = 0;
435 }
436
Igor Sysoevc7f876b2009-04-16 19:25:09 +0000437 if (shm_zone[i].shm.name.len != oshm_zone[n].shm.name.len) {
Igor Sysoev67cd3362006-11-20 08:51:45 +0000438 continue;
439 }
440
Igor Sysoevc7f876b2009-04-16 19:25:09 +0000441 if (ngx_strncmp(shm_zone[i].shm.name.data,
442 oshm_zone[n].shm.name.data,
443 shm_zone[i].shm.name.len)
Igor Sysoev28c7f762007-01-02 23:54:14 +0000444 != 0)
445 {
446 continue;
447 }
448
449 if (shm_zone[i].shm.size == oshm_zone[n].shm.size) {
450 shm_zone[i].shm.addr = oshm_zone[n].shm.addr;
Igor Sysoev4e77a2b2007-01-09 15:59:20 +0000451
452 if (shm_zone[i].init(&shm_zone[i], oshm_zone[n].data)
453 != NGX_OK)
454 {
455 goto failed;
456 }
457
Igor Sysoev667a79b2007-01-09 16:02:29 +0000458 goto shm_zone_found;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000459 }
460
Igor Sysoev28c7f762007-01-02 23:54:14 +0000461 ngx_shm_free(&oshm_zone[n].shm);
Igor Sysoev67cd3362006-11-20 08:51:45 +0000462
463 break;
464 }
465
Igor Sysoev28c7f762007-01-02 23:54:14 +0000466 if (ngx_shm_alloc(&shm_zone[i].shm) != NGX_OK) {
Igor Sysoev67cd3362006-11-20 08:51:45 +0000467 goto failed;
468 }
469
Igor Sysoeva1195842009-06-02 13:57:59 +0000470 if (ngx_init_zone_pool(cycle, &shm_zone[i]) != NGX_OK) {
471 goto failed;
Igor Sysoev67cd3362006-11-20 08:51:45 +0000472 }
473
Igor Sysoev4e77a2b2007-01-09 15:59:20 +0000474 if (shm_zone[i].init(&shm_zone[i], NULL) != NGX_OK) {
Igor Sysoev28c7f762007-01-02 23:54:14 +0000475 goto failed;
476 }
477
Igor Sysoev667a79b2007-01-09 16:02:29 +0000478 shm_zone_found:
Igor Sysoev67cd3362006-11-20 08:51:45 +0000479
480 continue;
481 }
482
483
Igor Sysoevffe71442006-02-08 15:33:12 +0000484 /* handle the listening sockets */
Igor Sysoev1b735832004-11-11 14:07:14 +0000485
Igor Sysoevffe71442006-02-08 15:33:12 +0000486 if (old_cycle->listening.nelts) {
487 ls = old_cycle->listening.elts;
488 for (i = 0; i < old_cycle->listening.nelts; i++) {
489 ls[i].remain = 0;
490 }
491
492 nls = cycle->listening.elts;
493 for (n = 0; n < cycle->listening.nelts; n++) {
494
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000495 for (i = 0; i < old_cycle->listening.nelts; i++) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000496 if (ls[i].ignore) {
497 continue;
498 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000499
Igor Sysoevffe71442006-02-08 15:33:12 +0000500 if (ngx_cmp_sockaddr(nls[n].sockaddr, ls[i].sockaddr) == NGX_OK)
501 {
502 nls[n].fd = ls[i].fd;
503 nls[n].previous = &ls[i];
504 ls[i].remain = 1;
Igor Sysoev1b735832004-11-11 14:07:14 +0000505
Igor Sysoevffe71442006-02-08 15:33:12 +0000506 if (ls[n].backlog != nls[i].backlog) {
507 nls[n].listen = 1;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000508 }
509
Igor Sysoevb145b062005-06-15 18:33:41 +0000510#if (NGX_HAVE_DEFERRED_ACCEPT && defined SO_ACCEPTFILTER)
511
Igor Sysoevffe71442006-02-08 15:33:12 +0000512 /*
513 * FreeBSD, except the most recent versions,
514 * could not remove accept filter
515 */
516 nls[n].deferred_accept = ls[i].deferred_accept;
Igor Sysoevb145b062005-06-15 18:33:41 +0000517
Igor Sysoevffe71442006-02-08 15:33:12 +0000518 if (ls[i].accept_filter && nls[n].accept_filter) {
519 if (ngx_strcmp(ls[i].accept_filter,
520 nls[n].accept_filter)
521 != 0)
Igor Sysoevb145b062005-06-15 18:33:41 +0000522 {
Igor Sysoevffe71442006-02-08 15:33:12 +0000523 nls[n].delete_deferred = 1;
Igor Sysoevb145b062005-06-15 18:33:41 +0000524 nls[n].add_deferred = 1;
525 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000526
Igor Sysoevffe71442006-02-08 15:33:12 +0000527 } else if (ls[i].accept_filter) {
528 nls[n].delete_deferred = 1;
529
530 } else if (nls[n].accept_filter) {
531 nls[n].add_deferred = 1;
532 }
533#endif
534
535#if (NGX_HAVE_DEFERRED_ACCEPT && defined TCP_DEFER_ACCEPT)
536
537 if (ls[n].deferred_accept && !nls[n].deferred_accept) {
538 nls[n].delete_deferred = 1;
539
540 } else if (ls[i].deferred_accept != nls[n].deferred_accept)
541 {
542 nls[n].add_deferred = 1;
543 }
544#endif
545 break;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000546 }
547 }
548
Igor Sysoevffe71442006-02-08 15:33:12 +0000549 if (nls[n].fd == -1) {
550 nls[n].open = 1;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000551 }
552 }
553
Igor Sysoevffe71442006-02-08 15:33:12 +0000554 } else {
555 ls = cycle->listening.elts;
556 for (i = 0; i < cycle->listening.nelts; i++) {
557 ls[i].open = 1;
558#if (NGX_HAVE_DEFERRED_ACCEPT && defined SO_ACCEPTFILTER)
559 if (ls[i].accept_filter) {
560 ls[i].add_deferred = 1;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000561 }
Igor Sysoevffe71442006-02-08 15:33:12 +0000562#endif
563#if (NGX_HAVE_DEFERRED_ACCEPT && defined TCP_DEFER_ACCEPT)
564 if (ls[i].deferred_accept) {
565 ls[i].add_deferred = 1;
Igor Sysoevb145b062005-06-15 18:33:41 +0000566 }
Igor Sysoevffe71442006-02-08 15:33:12 +0000567#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000568 }
569 }
570
Igor Sysoev165aa392009-06-06 12:41:31 +0000571 if (ngx_open_listening_sockets(cycle) != NGX_OK) {
572 goto failed;
573 }
Igor Sysoev9e580192006-02-01 18:22:15 +0000574
Igor Sysoev165aa392009-06-06 12:41:31 +0000575 if (!ngx_test_config) {
576 ngx_configure_listening_sockets(cycle);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000577 }
578
Igor Sysoev25b36fe2004-02-03 16:43:54 +0000579
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000580 /* commit the new cycle configuration */
581
Igor Sysoevc28ff712009-04-23 11:13:12 +0000582 if (!ngx_use_stderr && cycle->log->file->fd != ngx_stderr) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000583
Igor Sysoevc28ff712009-04-23 11:13:12 +0000584 if (ngx_set_stderr(cycle->log->file->fd) == NGX_FILE_ERROR) {
585 ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
586 ngx_set_stderr_n " failed");
Igor Sysoevaad1b892004-10-03 20:02:06 +0000587 }
588 }
589
Igor Sysoevaad1b892004-10-03 20:02:06 +0000590 pool->log = cycle->log;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000591
592 for (i = 0; ngx_modules[i]; i++) {
593 if (ngx_modules[i]->init_module) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000594 if (ngx_modules[i]->init_module(cycle) != NGX_OK) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000595 /* fatal */
596 exit(1);
597 }
598 }
599 }
600
Igor Sysoevd43bee82004-11-20 19:52:20 +0000601
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000602 /* close and delete stuff that lefts from an old cycle */
603
Igor Sysoev3a257b32007-01-09 16:00:57 +0000604 /* free the unnecessary shared memory */
605
606 opart = &old_cycle->shared_memory.part;
607 oshm_zone = opart->elts;
608
609 for (i = 0; /* void */ ; i++) {
610
611 if (i >= opart->nelts) {
612 if (opart->next == NULL) {
613 goto old_shm_zone_done;
614 }
615 opart = opart->next;
616 oshm_zone = opart->elts;
617 i = 0;
618 }
619
620 part = &cycle->shared_memory.part;
621 shm_zone = part->elts;
622
623 for (n = 0; /* void */ ; n++) {
624
625 if (n >= part->nelts) {
626 if (part->next == NULL) {
627 break;
628 }
629 part = part->next;
630 shm_zone = part->elts;
631 n = 0;
632 }
633
Igor Sysoevc7f876b2009-04-16 19:25:09 +0000634 if (oshm_zone[i].shm.name.len == shm_zone[n].shm.name.len
635 && ngx_strncmp(oshm_zone[i].shm.name.data,
636 shm_zone[n].shm.name.data,
637 oshm_zone[i].shm.name.len)
Igor Sysoev3a257b32007-01-09 16:00:57 +0000638 == 0)
639 {
640 goto live_shm_zone;
641 }
642 }
643
644 ngx_shm_free(&oshm_zone[i].shm);
645
646 live_shm_zone:
647
648 continue;
649 }
650
651old_shm_zone_done:
652
653
Igor Sysoeva269c822007-01-09 16:00:07 +0000654 /* close the unnecessary listening sockets */
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000655
656 ls = old_cycle->listening.elts;
657 for (i = 0; i < old_cycle->listening.nelts; i++) {
Igor Sysoev366bf712009-06-06 12:53:55 +0000658
659 if (ls[i].remain || ls[i].fd == -1) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000660 continue;
661 }
662
663 if (ngx_close_socket(ls[i].fd) == -1) {
664 ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno,
Igor Sysoev31eb8c02005-09-23 11:02:22 +0000665 ngx_close_socket_n " listening socket on %V failed",
Igor Sysoev1b735832004-11-11 14:07:14 +0000666 &ls[i].addr_text);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000667 }
Igor Sysoev1f073922010-06-04 13:34:23 +0000668
669#if (NGX_HAVE_UNIX_DOMAIN)
670
671 if (ls[i].sockaddr->sa_family == AF_UNIX) {
672 u_char *name;
673
674 name = ls[i].addr_text.data + sizeof("unix:") - 1;
675
676 ngx_log_error(NGX_LOG_WARN, cycle->log, 0,
677 "deleting socket %s", name);
678
679 if (ngx_delete_file(name) == -1) {
680 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_socket_errno,
681 ngx_delete_file_n " %s failed", name);
682 }
683 }
684
685#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000686 }
687
688
Igor Sysoeva269c822007-01-09 16:00:07 +0000689 /* close the unnecessary open files */
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000690
Igor Sysoevb9e34412004-09-03 15:50:30 +0000691 part = &old_cycle->open_files.part;
692 file = part->elts;
693
694 for (i = 0; /* void */ ; i++) {
695
696 if (i >= part->nelts) {
697 if (part->next == NULL) {
698 break;
699 }
700 part = part->next;
701 file = part->elts;
702 i = 0;
703 }
704
Igor Sysoevc28ff712009-04-23 11:13:12 +0000705 if (file[i].fd == NGX_INVALID_FILE || file[i].fd == ngx_stderr) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000706 continue;
707 }
708
709 if (ngx_close_file(file[i].fd) == NGX_FILE_ERROR) {
710 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
711 ngx_close_file_n " \"%s\" failed",
712 file[i].name.data);
713 }
714 }
715
Igor Sysoev305a9d82005-12-26 17:07:48 +0000716 ngx_destroy_pool(conf.temp_pool);
717
Igor Sysoev9e580192006-02-01 18:22:15 +0000718 if (ngx_process == NGX_PROCESS_MASTER || ngx_is_init_cycle(old_cycle)) {
719
Igor Sysoev7f35ae62007-12-16 11:58:16 +0000720 /*
Igor Sysoev2d95c3f2009-09-18 09:12:40 +0000721 * perl_destruct() frees environ, if it is not the same as it was at
722 * perl_construct() time, therefore we save the previous cycle
Igor Sysoev7f35ae62007-12-16 11:58:16 +0000723 * environment before ngx_conf_parse() where it will be changed.
724 */
725
726 env = environ;
727 environ = senv;
728
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000729 ngx_destroy_pool(old_cycle->pool);
Igor Sysoevffe71442006-02-08 15:33:12 +0000730 cycle->old_cycle = NULL;
731
Igor Sysoev7f35ae62007-12-16 11:58:16 +0000732 environ = env;
733
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000734 return cycle;
735 }
736
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000737
738 if (ngx_temp_pool == NULL) {
739 ngx_temp_pool = ngx_create_pool(128, cycle->log);
740 if (ngx_temp_pool == NULL) {
741 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
Ruslan Ermilova823c552011-09-19 14:48:29 +0000742 "could not create ngx_temp_pool");
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000743 exit(1);
744 }
745
746 n = 10;
747 ngx_old_cycles.elts = ngx_pcalloc(ngx_temp_pool,
748 n * sizeof(ngx_cycle_t *));
749 if (ngx_old_cycles.elts == NULL) {
750 exit(1);
751 }
752 ngx_old_cycles.nelts = 0;
753 ngx_old_cycles.size = sizeof(ngx_cycle_t *);
754 ngx_old_cycles.nalloc = n;
755 ngx_old_cycles.pool = ngx_temp_pool;
756
Igor Sysoev899b44e2005-05-12 14:58:06 +0000757 ngx_cleaner_event.handler = ngx_clean_old_cycles;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000758 ngx_cleaner_event.log = cycle->log;
759 ngx_cleaner_event.data = &dumb;
760 dumb.fd = (ngx_socket_t) -1;
761 }
762
763 ngx_temp_pool->log = cycle->log;
764
Igor Sysoevc1571722005-03-19 12:38:37 +0000765 old = ngx_array_push(&ngx_old_cycles);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000766 if (old == NULL) {
767 exit(1);
768 }
769 *old = old_cycle;
770
771 if (!ngx_cleaner_event.timer_set) {
772 ngx_add_timer(&ngx_cleaner_event, 30000);
773 ngx_cleaner_event.timer_set = 1;
774 }
775
776 return cycle;
Igor Sysoevffe71442006-02-08 15:33:12 +0000777
778
779failed:
780
Igor Sysoeve5e4c002007-04-18 11:28:11 +0000781 if (!ngx_is_init_cycle(old_cycle)) {
782 old_ccf = (ngx_core_conf_t *) ngx_get_conf(old_cycle->conf_ctx,
783 ngx_core_module);
784 if (old_ccf->environment) {
785 environ = old_ccf->environment;
786 }
787 }
788
Igor Sysoevffe71442006-02-08 15:33:12 +0000789 /* rollback the new cycle configuration */
790
791 part = &cycle->open_files.part;
792 file = part->elts;
793
794 for (i = 0; /* void */ ; i++) {
795
796 if (i >= part->nelts) {
797 if (part->next == NULL) {
798 break;
799 }
800 part = part->next;
801 file = part->elts;
802 i = 0;
803 }
804
Igor Sysoevc28ff712009-04-23 11:13:12 +0000805 if (file[i].fd == NGX_INVALID_FILE || file[i].fd == ngx_stderr) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000806 continue;
807 }
808
809 if (ngx_close_file(file[i].fd) == NGX_FILE_ERROR) {
810 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
811 ngx_close_file_n " \"%s\" failed",
812 file[i].name.data);
813 }
814 }
815
816 if (ngx_test_config) {
817 ngx_destroy_cycle_pools(&conf);
818 return NULL;
819 }
820
821 ls = cycle->listening.elts;
822 for (i = 0; i < cycle->listening.nelts; i++) {
823 if (ls[i].fd == -1 || !ls[i].open) {
824 continue;
825 }
826
827 if (ngx_close_socket(ls[i].fd) == -1) {
828 ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno,
829 ngx_close_socket_n " %V failed",
830 &ls[i].addr_text);
831 }
832 }
833
834 ngx_destroy_cycle_pools(&conf);
835
836 return NULL;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000837}
838
839
Igor Sysoev9e580192006-02-01 18:22:15 +0000840static void
841ngx_destroy_cycle_pools(ngx_conf_t *conf)
842{
Igor Sysoev9e580192006-02-01 18:22:15 +0000843 ngx_destroy_pool(conf->temp_pool);
844 ngx_destroy_pool(conf->pool);
845}
846
847
Igor Sysoevc2068d02005-10-19 12:33:58 +0000848static ngx_int_t
849ngx_cmp_sockaddr(struct sockaddr *sa1, struct sockaddr *sa2)
Igor Sysoev1b735832004-11-11 14:07:14 +0000850{
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000851 struct sockaddr_in *sin1, *sin2;
852#if (NGX_HAVE_INET6)
853 struct sockaddr_in6 *sin61, *sin62;
854#endif
Igor Sysoev1f073922010-06-04 13:34:23 +0000855#if (NGX_HAVE_UNIX_DOMAIN)
856 struct sockaddr_un *saun1, *saun2;
857#endif
Igor Sysoev1b735832004-11-11 14:07:14 +0000858
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000859 if (sa1->sa_family != sa2->sa_family) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000860 return NGX_DECLINED;
861 }
862
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000863 switch (sa1->sa_family) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000864
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000865#if (NGX_HAVE_INET6)
866 case AF_INET6:
867 sin61 = (struct sockaddr_in6 *) sa1;
868 sin62 = (struct sockaddr_in6 *) sa2;
Igor Sysoev1b735832004-11-11 14:07:14 +0000869
Igor Sysoevce03ba02010-06-30 12:01:56 +0000870 if (sin61->sin6_port != sin62->sin6_port) {
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000871 return NGX_DECLINED;
872 }
873
874 if (ngx_memcmp(&sin61->sin6_addr, &sin62->sin6_addr, 16) != 0) {
875 return NGX_DECLINED;
876 }
877
878 break;
879#endif
880
Igor Sysoev1f073922010-06-04 13:34:23 +0000881#if (NGX_HAVE_UNIX_DOMAIN)
882 case AF_UNIX:
883 saun1 = (struct sockaddr_un *) sa1;
884 saun2 = (struct sockaddr_un *) sa2;
885
886 if (ngx_memcmp(&saun1->sun_path, &saun2->sun_path,
887 sizeof(saun1->sun_path))
888 != 0)
889 {
890 return NGX_DECLINED;
891 }
892
893 break;
894#endif
895
Igor Sysoeva35eacc2009-02-21 07:02:02 +0000896 default: /* AF_INET */
897
898 sin1 = (struct sockaddr_in *) sa1;
899 sin2 = (struct sockaddr_in *) sa2;
900
901 if (sin1->sin_port != sin2->sin_port) {
902 return NGX_DECLINED;
903 }
904
905 if (sin1->sin_addr.s_addr != sin2->sin_addr.s_addr) {
906 return NGX_DECLINED;
907 }
908
909 break;
Igor Sysoev1b735832004-11-11 14:07:14 +0000910 }
911
912 return NGX_OK;
913}
914
915
Igor Sysoevf7a08d52009-04-18 19:27:28 +0000916static ngx_int_t
917ngx_init_zone_pool(ngx_cycle_t *cycle, ngx_shm_zone_t *zn)
918{
919 u_char *file;
920 ngx_slab_pool_t *sp;
921
922 sp = (ngx_slab_pool_t *) zn->shm.addr;
923
Igor Sysoeva1195842009-06-02 13:57:59 +0000924 if (zn->shm.exists) {
925
926 if (sp == sp->addr) {
927 return NGX_OK;
928 }
929
930 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
931 "shared zone \"%V\" has no equal addresses: %p vs %p",
932 &zn->shm.name, sp->addr, sp);
933 return NGX_ERROR;
934 }
935
Igor Sysoevf7a08d52009-04-18 19:27:28 +0000936 sp->end = zn->shm.addr + zn->shm.size;
937 sp->min_shift = 3;
Igor Sysoeva1195842009-06-02 13:57:59 +0000938 sp->addr = zn->shm.addr;
Igor Sysoevf7a08d52009-04-18 19:27:28 +0000939
940#if (NGX_HAVE_ATOMIC_OPS)
941
942 file = NULL;
943
944#else
945
946 file = ngx_pnalloc(cycle->pool, cycle->lock_file.len + zn->shm.name.len);
947 if (file == NULL) {
Igor Sysoevdc5ad522009-04-21 19:38:02 +0000948 return NGX_ERROR;
Igor Sysoevf7a08d52009-04-18 19:27:28 +0000949 }
950
951 (void) ngx_sprintf(file, "%V%V%Z", &cycle->lock_file, &zn->shm.name);
952
953#endif
954
Maxim Dounin91ecc8f2011-11-23 13:55:38 +0000955 if (ngx_shmtx_create(&sp->mutex, &sp->lock, file) != NGX_OK) {
Igor Sysoevdc5ad522009-04-21 19:38:02 +0000956 return NGX_ERROR;
Igor Sysoevf7a08d52009-04-18 19:27:28 +0000957 }
958
959 ngx_slab_init(sp);
960
961 return NGX_OK;
962}
963
964
Igor Sysoevc2068d02005-10-19 12:33:58 +0000965ngx_int_t
Igor Sysoevffe71442006-02-08 15:33:12 +0000966ngx_create_pidfile(ngx_str_t *name, ngx_log_t *log)
Igor Sysoev076498e2004-04-12 06:10:53 +0000967{
Igor Sysoevb37316d2009-04-08 19:51:30 +0000968 size_t len;
969 ngx_uint_t create;
970 ngx_file_t file;
971 u_char pid[NGX_INT64_LEN + 2];
Igor Sysoev076498e2004-04-12 06:10:53 +0000972
Igor Sysoevbd919992009-04-20 06:08:47 +0000973 if (ngx_process > NGX_PROCESS_MASTER) {
974 return NGX_OK;
975 }
976
Igor Sysoev43f13192004-04-12 16:38:09 +0000977 ngx_memzero(&file, sizeof(ngx_file_t));
Igor Sysoev31eb8c02005-09-23 11:02:22 +0000978
Igor Sysoevffe71442006-02-08 15:33:12 +0000979 file.name = *name;
980 file.log = log;
Igor Sysoev076498e2004-04-12 06:10:53 +0000981
Igor Sysoev2ec2e352009-04-08 19:03:41 +0000982 create = ngx_test_config ? NGX_FILE_CREATE_OR_OPEN : NGX_FILE_TRUNCATE;
Igor Sysoev090849d2004-05-18 20:28:54 +0000983
Igor Sysoev43f13192004-04-12 16:38:09 +0000984 file.fd = ngx_open_file(file.name.data, NGX_FILE_RDWR,
Igor Sysoev2ec2e352009-04-08 19:03:41 +0000985 create, NGX_FILE_DEFAULT_ACCESS);
Igor Sysoev43f13192004-04-12 16:38:09 +0000986
987 if (file.fd == NGX_INVALID_FILE) {
Igor Sysoevffe71442006-02-08 15:33:12 +0000988 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
Igor Sysoev43f13192004-04-12 16:38:09 +0000989 ngx_open_file_n " \"%s\" failed", file.name.data);
Igor Sysoev076498e2004-04-12 06:10:53 +0000990 return NGX_ERROR;
991 }
992
Igor Sysoev090849d2004-05-18 20:28:54 +0000993 if (!ngx_test_config) {
Igor Sysoev9ba14ac2006-12-22 20:30:26 +0000994 len = ngx_snprintf(pid, NGX_INT64_LEN + 2, "%P%N", ngx_pid) - pid;
Igor Sysoev1b735832004-11-11 14:07:14 +0000995
Igor Sysoev090849d2004-05-18 20:28:54 +0000996 if (ngx_write_file(&file, pid, len, 0) == NGX_ERROR) {
997 return NGX_ERROR;
998 }
Igor Sysoev076498e2004-04-12 06:10:53 +0000999 }
1000
Igor Sysoev43f13192004-04-12 16:38:09 +00001001 if (ngx_close_file(file.fd) == NGX_FILE_ERROR) {
Igor Sysoevffe71442006-02-08 15:33:12 +00001002 ngx_log_error(NGX_LOG_ALERT, log, ngx_errno,
Igor Sysoev43f13192004-04-12 16:38:09 +00001003 ngx_close_file_n " \"%s\" failed", file.name.data);
Igor Sysoev076498e2004-04-12 06:10:53 +00001004 }
1005
1006 return NGX_OK;
1007}
1008
1009
Igor Sysoevc2068d02005-10-19 12:33:58 +00001010void
1011ngx_delete_pidfile(ngx_cycle_t *cycle)
Igor Sysoev0e5dc5c2005-11-15 13:30:52 +00001012{
Igor Sysoev076498e2004-04-12 06:10:53 +00001013 u_char *name;
1014 ngx_core_conf_t *ccf;
1015
1016 ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
1017
Igor Sysoev31eb8c02005-09-23 11:02:22 +00001018 name = ngx_new_binary ? ccf->oldpid.data : ccf->pid.data;
Igor Sysoev076498e2004-04-12 06:10:53 +00001019
1020 if (ngx_delete_file(name) == NGX_FILE_ERROR) {
1021 ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
1022 ngx_delete_file_n " \"%s\" failed", name);
1023 }
1024}
1025
1026
Igor Sysoevc8e9f262009-04-21 20:25:49 +00001027ngx_int_t
1028ngx_signal_process(ngx_cycle_t *cycle, char *sig)
1029{
1030 ssize_t n;
1031 ngx_int_t pid;
1032 ngx_file_t file;
1033 ngx_core_conf_t *ccf;
1034 u_char buf[NGX_INT64_LEN + 2];
1035
1036 ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "signal process started");
1037
1038 ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
1039
1040 file.name = ccf->pid;
1041 file.log = cycle->log;
1042
1043 file.fd = ngx_open_file(file.name.data, NGX_FILE_RDONLY,
1044 NGX_FILE_OPEN, NGX_FILE_DEFAULT_ACCESS);
1045
1046 if (file.fd == NGX_INVALID_FILE) {
1047 ngx_log_error(NGX_LOG_ERR, cycle->log, ngx_errno,
1048 ngx_open_file_n " \"%s\" failed", file.name.data);
1049 return 1;
1050 }
1051
1052 n = ngx_read_file(&file, buf, NGX_INT64_LEN + 2, 0);
1053
1054 if (ngx_close_file(file.fd) == NGX_FILE_ERROR) {
1055 ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
1056 ngx_close_file_n " \"%s\" failed", file.name.data);
1057 }
1058
1059 if (n == NGX_ERROR) {
1060 return 1;
1061 }
1062
1063 while (n-- && (buf[n] == CR || buf[n] == LF)) { /* void */ }
1064
1065 pid = ngx_atoi(buf, ++n);
1066
1067 if (pid == NGX_ERROR) {
1068 ngx_log_error(NGX_LOG_ERR, cycle->log, 0,
1069 "invalid PID number \"%*s\" in \"%s\"",
1070 n, buf, file.name.data);
1071 return 1;
1072 }
1073
1074 return ngx_os_signal_process(cycle, sig, pid);
1075
1076}
1077
1078
Igor Sysoevffe71442006-02-08 15:33:12 +00001079static ngx_int_t
1080ngx_test_lockfile(u_char *file, ngx_log_t *log)
1081{
1082#if !(NGX_HAVE_ATOMIC_OPS)
1083 ngx_fd_t fd;
1084
Igor Sysoev8e750112007-02-11 07:49:12 +00001085 fd = ngx_open_file(file, NGX_FILE_RDWR, NGX_FILE_CREATE_OR_OPEN,
1086 NGX_FILE_DEFAULT_ACCESS);
Igor Sysoevffe71442006-02-08 15:33:12 +00001087
1088 if (fd == NGX_INVALID_FILE) {
1089 ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
1090 ngx_open_file_n " \"%s\" failed", file);
1091 return NGX_ERROR;
1092 }
1093
1094 if (ngx_close_file(fd) == NGX_FILE_ERROR) {
1095 ngx_log_error(NGX_LOG_ALERT, log, ngx_errno,
1096 ngx_close_file_n " \"%s\" failed", file);
1097 }
1098
1099 if (ngx_delete_file(file) == NGX_FILE_ERROR) {
1100 ngx_log_error(NGX_LOG_ALERT, log, ngx_errno,
1101 ngx_delete_file_n " \"%s\" failed", file);
1102 }
1103
1104#endif
1105
1106 return NGX_OK;
1107}
1108
1109
Igor Sysoevc2068d02005-10-19 12:33:58 +00001110void
1111ngx_reopen_files(ngx_cycle_t *cycle, ngx_uid_t user)
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001112{
Igor Sysoevf2884e12008-11-11 16:17:45 +00001113 ssize_t n, len;
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001114 ngx_fd_t fd;
Igor Sysoev10a543a2004-03-16 07:10:12 +00001115 ngx_uint_t i;
Igor Sysoevb9e34412004-09-03 15:50:30 +00001116 ngx_list_part_t *part;
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001117 ngx_open_file_t *file;
1118
Igor Sysoevb9e34412004-09-03 15:50:30 +00001119 part = &cycle->open_files.part;
1120 file = part->elts;
1121
1122 for (i = 0; /* void */ ; i++) {
1123
1124 if (i >= part->nelts) {
1125 if (part->next == NULL) {
1126 break;
1127 }
1128 part = part->next;
Igor Sysoevb145b062005-06-15 18:33:41 +00001129 file = part->elts;
Igor Sysoevb9e34412004-09-03 15:50:30 +00001130 i = 0;
1131 }
1132
Igor Sysoev5ef370d2009-04-27 11:32:33 +00001133 if (file[i].name.len == 0) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001134 continue;
1135 }
1136
Igor Sysoevf2884e12008-11-11 16:17:45 +00001137 len = file[i].pos - file[i].buffer;
1138
1139 if (file[i].buffer && len != 0) {
1140
1141 n = ngx_write_fd(file[i].fd, file[i].buffer, len);
1142
1143 if (n == NGX_FILE_ERROR) {
1144 ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
1145 ngx_write_fd_n " to \"%s\" failed",
1146 file[i].name.data);
1147
1148 } else if (n != len) {
1149 ngx_log_error(NGX_LOG_ALERT, cycle->log, 0,
1150 ngx_write_fd_n " to \"%s\" was incomplete: %z of %uz",
1151 file[i].name.data, n, len);
1152 }
1153
Igor Sysoev697d1ae2005-10-27 15:46:13 +00001154 file[i].pos = file[i].buffer;
1155 }
1156
Igor Sysoev24c27872009-03-31 13:52:01 +00001157 fd = ngx_open_file(file[i].name.data, NGX_FILE_APPEND,
Igor Sysoevfadc7a72009-03-30 14:51:51 +00001158 NGX_FILE_CREATE_OR_OPEN, NGX_FILE_DEFAULT_ACCESS);
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001159
1160 ngx_log_debug3(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
1161 "reopen file \"%s\", old:%d new:%d",
1162 file[i].name.data, file[i].fd, fd);
1163
1164 if (fd == NGX_INVALID_FILE) {
1165 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1166 ngx_open_file_n " \"%s\" failed", file[i].name.data);
1167 continue;
1168 }
1169
Igor Sysoeva2de7b92009-04-08 19:13:28 +00001170#if !(NGX_WIN32)
Igor Sysoev4959ec42005-05-23 12:07:45 +00001171 if (user != (ngx_uid_t) NGX_CONF_UNSET_UINT) {
Igor Sysoev86ef6aa2007-12-10 12:09:51 +00001172 ngx_file_info_t fi;
Igor Sysoev924bd792004-10-11 15:07:03 +00001173
Igor Sysoevef919752009-04-29 19:28:52 +00001174 if (ngx_file_info((const char *) file[i].name.data, &fi)
1175 == NGX_FILE_ERROR)
1176 {
Igor Sysoev924bd792004-10-11 15:07:03 +00001177 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1178 ngx_file_info_n " \"%s\" failed",
1179 file[i].name.data);
1180
1181 if (ngx_close_file(fd) == NGX_FILE_ERROR) {
1182 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1183 ngx_close_file_n " \"%s\" failed",
1184 file[i].name.data);
1185 }
1186 }
1187
Igor Sysoevd43bee82004-11-20 19:52:20 +00001188 if (fi.st_uid != user) {
1189 if (chown((const char *) file[i].name.data, user, -1) == -1) {
1190 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1191 "chown(\"%s\", %d) failed",
1192 file[i].name.data, user);
1193
1194 if (ngx_close_file(fd) == NGX_FILE_ERROR) {
1195 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1196 ngx_close_file_n " \"%s\" failed",
1197 file[i].name.data);
1198 }
1199 }
1200 }
1201
Igor Sysoev924bd792004-10-11 15:07:03 +00001202 if ((fi.st_mode & (S_IRUSR|S_IWUSR)) != (S_IRUSR|S_IWUSR)) {
1203
1204 fi.st_mode |= (S_IRUSR|S_IWUSR);
1205
1206 if (chmod((const char *) file[i].name.data, fi.st_mode) == -1) {
1207 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
Igor Sysoevd43bee82004-11-20 19:52:20 +00001208 "chmod() \"%s\" failed", file[i].name.data);
Igor Sysoev924bd792004-10-11 15:07:03 +00001209
1210 if (ngx_close_file(fd) == NGX_FILE_ERROR) {
1211 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1212 ngx_close_file_n " \"%s\" failed",
1213 file[i].name.data);
1214 }
1215 }
1216 }
Igor Sysoeva5362982004-03-04 07:04:55 +00001217 }
1218
Igor Sysoev0911f772004-01-14 18:19:42 +00001219 if (fcntl(fd, F_SETFD, FD_CLOEXEC) == -1) {
1220 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1221 "fcntl(FD_CLOEXEC) \"%s\" failed",
1222 file[i].name.data);
1223
1224 if (ngx_close_file(fd) == NGX_FILE_ERROR) {
1225 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1226 ngx_close_file_n " \"%s\" failed",
1227 file[i].name.data);
1228 }
1229
1230 continue;
1231 }
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001232#endif
1233
1234 if (ngx_close_file(file[i].fd) == NGX_FILE_ERROR) {
1235 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1236 ngx_close_file_n " \"%s\" failed",
1237 file[i].name.data);
1238 }
1239
1240 file[i].fd = fd;
1241 }
Igor Sysoev25b36fe2004-02-03 16:43:54 +00001242
Igor Sysoev1b735832004-11-11 14:07:14 +00001243#if !(NGX_WIN32)
Igor Sysoeva893eab2004-03-11 21:34:52 +00001244
Igor Sysoev980a9242004-09-05 19:54:02 +00001245 if (cycle->log->file->fd != STDERR_FILENO) {
1246 if (dup2(cycle->log->file->fd, STDERR_FILENO) == -1) {
1247 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
1248 "dup2(STDERR) failed");
1249 }
Igor Sysoev25b36fe2004-02-03 16:43:54 +00001250 }
Igor Sysoeva893eab2004-03-11 21:34:52 +00001251
Igor Sysoeva5362982004-03-04 07:04:55 +00001252#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001253}
1254
1255
Igor Sysoev28c7f762007-01-02 23:54:14 +00001256ngx_shm_zone_t *
1257ngx_shared_memory_add(ngx_conf_t *cf, ngx_str_t *name, size_t size, void *tag)
1258{
1259 ngx_uint_t i;
1260 ngx_shm_zone_t *shm_zone;
1261 ngx_list_part_t *part;
1262
1263 part = &cf->cycle->shared_memory.part;
1264 shm_zone = part->elts;
1265
1266 for (i = 0; /* void */ ; i++) {
1267
1268 if (i >= part->nelts) {
1269 if (part->next == NULL) {
1270 break;
1271 }
1272 part = part->next;
1273 shm_zone = part->elts;
1274 i = 0;
1275 }
1276
Igor Sysoevc7f876b2009-04-16 19:25:09 +00001277 if (name->len != shm_zone[i].shm.name.len) {
Igor Sysoev28c7f762007-01-02 23:54:14 +00001278 continue;
1279 }
1280
Igor Sysoevc7f876b2009-04-16 19:25:09 +00001281 if (ngx_strncmp(name->data, shm_zone[i].shm.name.data, name->len)
1282 != 0)
1283 {
Igor Sysoev28c7f762007-01-02 23:54:14 +00001284 continue;
1285 }
1286
1287 if (size && size != shm_zone[i].shm.size) {
1288 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
Igor Sysoevc7f876b2009-04-16 19:25:09 +00001289 "the size %uz of shared memory zone \"%V\" "
1290 "conflicts with already declared size %uz",
1291 size, &shm_zone[i].shm.name, shm_zone[i].shm.size);
Igor Sysoev28c7f762007-01-02 23:54:14 +00001292 return NULL;
1293 }
1294
1295 if (tag != shm_zone[i].tag) {
1296 ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
Igor Sysoevc7f876b2009-04-16 19:25:09 +00001297 "the shared memory zone \"%V\" is "
1298 "already declared for a different use",
1299 &shm_zone[i].shm.name);
Igor Sysoev28c7f762007-01-02 23:54:14 +00001300 return NULL;
1301 }
1302
1303 return &shm_zone[i];
1304 }
1305
1306 shm_zone = ngx_list_push(&cf->cycle->shared_memory);
1307
1308 if (shm_zone == NULL) {
1309 return NULL;
1310 }
1311
1312 shm_zone->data = NULL;
1313 shm_zone->shm.log = cf->cycle->log;
1314 shm_zone->shm.size = size;
Igor Sysoevc7f876b2009-04-16 19:25:09 +00001315 shm_zone->shm.name = *name;
Igor Sysoevf7a08d52009-04-18 19:27:28 +00001316 shm_zone->shm.exists = 0;
Igor Sysoev28c7f762007-01-02 23:54:14 +00001317 shm_zone->init = NULL;
Igor Sysoev28c7f762007-01-02 23:54:14 +00001318 shm_zone->tag = tag;
1319
1320 return shm_zone;
1321}
1322
1323
Igor Sysoevc2068d02005-10-19 12:33:58 +00001324static void
1325ngx_clean_old_cycles(ngx_event_t *ev)
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001326{
Igor Sysoev10a543a2004-03-16 07:10:12 +00001327 ngx_uint_t i, n, found, live;
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001328 ngx_log_t *log;
1329 ngx_cycle_t **cycle;
1330
1331 log = ngx_cycle->log;
1332 ngx_temp_pool->log = log;
1333
Igor Sysoev54498db2004-02-11 17:08:49 +00001334 ngx_log_debug0(NGX_LOG_DEBUG_CORE, log, 0, "clean old cycles");
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001335
1336 live = 0;
1337
1338 cycle = ngx_old_cycles.elts;
1339 for (i = 0; i < ngx_old_cycles.nelts; i++) {
1340
1341 if (cycle[i] == NULL) {
1342 continue;
1343 }
1344
1345 found = 0;
1346
1347 for (n = 0; n < cycle[i]->connection_n; n++) {
Igor Sysoev78452232005-10-12 13:50:36 +00001348 if (cycle[i]->connections[n].fd != (ngx_socket_t) -1) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001349 found = 1;
Igor Sysoev54498db2004-02-11 17:08:49 +00001350
1351 ngx_log_debug1(NGX_LOG_DEBUG_CORE, log, 0, "live fd:%d", n);
1352
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001353 break;
1354 }
1355 }
1356
1357 if (found) {
1358 live = 1;
1359 continue;
1360 }
1361
Igor Sysoev54498db2004-02-11 17:08:49 +00001362 ngx_log_debug1(NGX_LOG_DEBUG_CORE, log, 0, "clean old cycle: %d", i);
1363
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001364 ngx_destroy_pool(cycle[i]->pool);
1365 cycle[i] = NULL;
1366 }
1367
Igor Sysoev54498db2004-02-11 17:08:49 +00001368 ngx_log_debug1(NGX_LOG_DEBUG_CORE, log, 0, "old cycles status: %d", live);
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001369
1370 if (live) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +00001371 ngx_add_timer(ev, 30000);
1372
1373 } else {
1374 ngx_destroy_pool(ngx_temp_pool);
1375 ngx_temp_pool = NULL;
1376 ngx_old_cycles.nelts = 0;
1377 }
1378}