blob: 7b98e1263650883f3f1a96ec6e4fe278cffe1981 [file] [log] [blame]
Igor Sysoev6de5c2c2002-08-06 16:39:45 +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 Sysoev6de5c2c2002-08-06 16:39:45 +00007#include <ngx_config.h>
Igor Sysoevb2620632003-01-10 06:09:20 +00008#include <ngx_core.h>
Igor Sysoev239baac2003-06-11 15:28:34 +00009#include <ngx_event.h>
Igor Sysoevaa3436c2003-05-30 14:27:59 +000010#include <nginx.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000011
12
Igor Sysoevfc5a10a2004-03-09 19:47:07 +000013static ngx_int_t ngx_add_inherited_sockets(ngx_cycle_t *cycle);
Igor Sysoev924bd792004-10-11 15:07:03 +000014static ngx_int_t ngx_getopt(ngx_cycle_t *cycle, int argc, char *const *argv);
15static ngx_int_t ngx_save_argv(ngx_cycle_t *cycle, int argc, char *const *argv);
Igor Sysoev43f13192004-04-12 16:38:09 +000016static void *ngx_core_module_create_conf(ngx_cycle_t *cycle);
17static char *ngx_core_module_init_conf(ngx_cycle_t *cycle, void *conf);
Igor Sysoev03420a62004-01-20 20:40:08 +000018static char *ngx_set_user(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
Igor Sysoev42b12b32004-12-02 18:40:46 +000019static char *ngx_set_priority(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
Igor Sysoev2b542382002-08-20 14:48:28 +000020
Igor Sysoev7f125082003-07-11 15:17:50 +000021
Igor Sysoev1b735832004-11-11 14:07:14 +000022static ngx_conf_enum_t ngx_debug_points[] = {
23 { ngx_string("stop"), NGX_DEBUG_POINTS_STOP },
24 { ngx_string("abort"), NGX_DEBUG_POINTS_ABORT },
25 { ngx_null_string, 0 }
26};
27
28
Igor Sysoev7f125082003-07-11 15:17:50 +000029static ngx_command_t ngx_core_commands[] = {
30
Igor Sysoev2b58fbf2003-12-09 15:08:11 +000031 { ngx_string("daemon"),
Igor Sysoev43f13192004-04-12 16:38:09 +000032 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
33 ngx_conf_set_flag_slot,
Igor Sysoev2b58fbf2003-12-09 15:08:11 +000034 0,
35 offsetof(ngx_core_conf_t, daemon),
36 NULL },
Igor Sysoev7f125082003-07-11 15:17:50 +000037
Igor Sysoevf0677992004-01-08 17:08:10 +000038 { ngx_string("master_process"),
Igor Sysoev43f13192004-04-12 16:38:09 +000039 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
40 ngx_conf_set_flag_slot,
Igor Sysoev3c3ca172004-01-05 20:55:48 +000041 0,
Igor Sysoevf0677992004-01-08 17:08:10 +000042 offsetof(ngx_core_conf_t, master),
Igor Sysoev3c3ca172004-01-05 20:55:48 +000043 NULL },
44
Igor Sysoeva741f8d2004-03-30 20:31:58 +000045 { ngx_string("worker_processes"),
Igor Sysoev43f13192004-04-12 16:38:09 +000046 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
47 ngx_conf_set_num_slot,
Igor Sysoeva741f8d2004-03-30 20:31:58 +000048 0,
49 offsetof(ngx_core_conf_t, worker_processes),
50 NULL },
51
Igor Sysoev1b735832004-11-11 14:07:14 +000052 { ngx_string("debug_points"),
53 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
54 ngx_conf_set_enum_slot,
55 0,
56 offsetof(ngx_core_conf_t, debug_points),
57 &ngx_debug_points },
58
Igor Sysoev32fcd5c2004-07-05 06:55:54 +000059#if (NGX_THREADS)
60
61 { ngx_string("worker_threads"),
62 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
63 ngx_conf_set_num_slot,
64 0,
65 offsetof(ngx_core_conf_t, worker_threads),
66 NULL },
67
68 { ngx_string("thread_stack_size"),
69 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
70 ngx_conf_set_size_slot,
71 0,
72 offsetof(ngx_core_conf_t, thread_stack_size),
73 NULL },
74
75#endif
76
Igor Sysoev43f13192004-04-12 16:38:09 +000077 { ngx_string("user"),
78 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE12,
79 ngx_set_user,
80 0,
81 0,
82 NULL },
83
Igor Sysoev42b12b32004-12-02 18:40:46 +000084 { ngx_string("worker_priority"),
85 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
86 ngx_set_priority,
87 0,
88 0,
89 NULL },
90
Igor Sysoeve9b2cb12004-02-09 20:47:18 +000091 { ngx_string("pid"),
Igor Sysoev43f13192004-04-12 16:38:09 +000092 NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1,
93 ngx_conf_set_str_slot,
Igor Sysoeve9b2cb12004-02-09 20:47:18 +000094 0,
95 offsetof(ngx_core_conf_t, pid),
96 NULL },
97
Igor Sysoev2b58fbf2003-12-09 15:08:11 +000098 ngx_null_command
Igor Sysoev7f125082003-07-11 15:17:50 +000099};
100
101
Igor Sysoev43f13192004-04-12 16:38:09 +0000102static ngx_core_module_t ngx_core_module_ctx = {
103 ngx_string("core"),
104 ngx_core_module_create_conf,
105 ngx_core_module_init_conf
106};
107
108
Igor Sysoev7f125082003-07-11 15:17:50 +0000109ngx_module_t ngx_core_module = {
Igor Sysoev899b44e2005-05-12 14:58:06 +0000110 NGX_MODULE_V1,
Igor Sysoev43f13192004-04-12 16:38:09 +0000111 &ngx_core_module_ctx, /* module context */
Igor Sysoev7f125082003-07-11 15:17:50 +0000112 ngx_core_commands, /* module directives */
113 NGX_CORE_MODULE, /* module type */
Igor Sysoev43f13192004-04-12 16:38:09 +0000114 NULL, /* init module */
Igor Sysoevaad1b892004-10-03 20:02:06 +0000115 NULL /* init process */
Igor Sysoev7f125082003-07-11 15:17:50 +0000116};
117
118
Igor Sysoev3f4685f2004-04-25 20:13:21 +0000119ngx_uint_t ngx_max_module;
Igor Sysoeva9830112003-05-19 16:39:14 +0000120
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000121
Igor Sysoev4d656dc2005-03-22 16:02:46 +0000122int ngx_cdecl
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000123main(int argc, char *const *argv)
Igor Sysoev6abfde62003-07-01 15:00:03 +0000124{
Igor Sysoev924bd792004-10-11 15:07:03 +0000125 ngx_int_t i;
126 ngx_log_t *log;
127 ngx_cycle_t *cycle, init_cycle;
128 ngx_core_conf_t *ccf;
Igor Sysoev6abfde62003-07-01 15:00:03 +0000129
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000130#if (NGX_FREEBSD)
Igor Sysoev62260f22003-12-05 17:07:27 +0000131 ngx_debug_init();
Igor Sysoev9d639522003-07-07 06:11:50 +0000132#endif
133
Igor Sysoevbc5c2872003-07-02 05:01:53 +0000134 /* TODO */ ngx_max_sockets = -1;
Igor Sysoev6abfde62003-07-01 15:00:03 +0000135
Igor Sysoeva8fa0a62003-11-25 20:44:56 +0000136 ngx_time_init();
Igor Sysoevf0677992004-01-08 17:08:10 +0000137
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000138#if (NGX_PCRE)
Igor Sysoeva8fa0a62003-11-25 20:44:56 +0000139 ngx_regex_init();
Igor Sysoeve89c4582003-12-19 08:15:11 +0000140#endif
Igor Sysoev562e53e2003-11-13 06:14:05 +0000141
Igor Sysoev25b36fe2004-02-03 16:43:54 +0000142 ngx_pid = ngx_getpid();
Igor Sysoev6abfde62003-07-01 15:00:03 +0000143
Igor Sysoevc1571722005-03-19 12:38:37 +0000144 log = ngx_log_init();
145 if (log == NULL) {
Igor Sysoevff8da912004-09-29 16:00:49 +0000146 return 1;
147 }
Igor Sysoev55168f62004-09-28 20:09:22 +0000148
Igor Sysoev1c3567e2004-07-15 16:35:51 +0000149#if (NGX_OPENSSL)
150 ngx_ssl_init(log);
151#endif
152
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000153 /* init_cycle->log is required for signal handlers and ngx_getopt() */
Igor Sysoeve89c4582003-12-19 08:15:11 +0000154
155 ngx_memzero(&init_cycle, sizeof(ngx_cycle_t));
156 init_cycle.log = log;
157 ngx_cycle = &init_cycle;
158
Igor Sysoevc1571722005-03-19 12:38:37 +0000159 init_cycle.pool = ngx_create_pool(1024, log);
160 if (init_cycle.pool == NULL) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000161 return 1;
162 }
163
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000164 if (ngx_save_argv(&init_cycle, argc, argv) == NGX_ERROR) {
Igor Sysoev924bd792004-10-11 15:07:03 +0000165 return 1;
166 }
167
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000168 if (ngx_getopt(&init_cycle, argc, ngx_argv) == NGX_ERROR) {
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000169 return 1;
170 }
171
Igor Sysoev8035fd22004-10-01 15:53:53 +0000172 if (ngx_test_config) {
173 log->log_level = NGX_LOG_INFO;
Igor Sysoev8035fd22004-10-01 15:53:53 +0000174 }
175
Igor Sysoev55168f62004-09-28 20:09:22 +0000176 if (ngx_os_init(log) == NGX_ERROR) {
177 return 1;
178 }
179
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000180 if (ngx_add_inherited_sockets(&init_cycle) == NGX_ERROR) {
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000181 return 1;
182 }
183
Igor Sysoev68df19d2004-04-15 15:34:36 +0000184 ngx_max_module = 0;
185 for (i = 0; ngx_modules[i]; i++) {
186 ngx_modules[i]->index = ngx_max_module++;
187 }
188
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000189 cycle = ngx_init_cycle(&init_cycle);
Igor Sysoevbc5c2872003-07-02 05:01:53 +0000190 if (cycle == NULL) {
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000191 if (ngx_test_config) {
192 ngx_log_error(NGX_LOG_EMERG, log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000193 "the configuration file \"%s\" test failed",
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000194 init_cycle.conf_file.data);
195 }
196
Igor Sysoev6abfde62003-07-01 15:00:03 +0000197 return 1;
198 }
199
Igor Sysoev68df19d2004-04-15 15:34:36 +0000200 if (ngx_test_config) {
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000201 ngx_log_error(NGX_LOG_INFO, log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000202 "the configuration file \"%s\" was tested successfully",
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000203 cycle->conf_file.data);
Igor Sysoev68df19d2004-04-15 15:34:36 +0000204 return 0;
205 }
206
Igor Sysoevaad1b892004-10-03 20:02:06 +0000207 ngx_os_status(cycle->log);
208
Igor Sysoev9d639522003-07-07 06:11:50 +0000209 ngx_cycle = cycle;
Igor Sysoev340b03b2003-07-04 15:10:33 +0000210
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000211 ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
212
Igor Sysoev43f13192004-04-12 16:38:09 +0000213 ngx_process = ccf->master ? NGX_PROCESS_MASTER : NGX_PROCESS_SINGLE;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000214
Igor Sysoev1b735832004-11-11 14:07:14 +0000215#if (NGX_WIN32)
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000216
217#if 0
Igor Sysoeve0207bb2004-06-23 15:18:17 +0000218
219 TODO:
220
Igor Sysoev43f13192004-04-12 16:38:09 +0000221 if (ccf->run_as_service) {
222 if (ngx_service(cycle->log) == NGX_ERROR) {
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000223 return 1;
224 }
225
226 return 0;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000227 }
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000228#endif
229
230#else
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000231
Igor Sysoev43f13192004-04-12 16:38:09 +0000232 if (!ngx_inherited && ccf->daemon) {
Igor Sysoev160d7742003-11-19 16:26:41 +0000233 if (ngx_daemon(cycle->log) == NGX_ERROR) {
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000234 return 1;
235 }
Igor Sysoeve0207bb2004-06-23 15:18:17 +0000236
237 ngx_daemonized = 1;
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000238 }
239
Igor Sysoev43f13192004-04-12 16:38:09 +0000240 if (ngx_create_pidfile(cycle, NULL) == NGX_ERROR) {
Igor Sysoevdc867cd2003-12-14 20:10:27 +0000241 return 1;
242 }
243
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000244#endif
Igor Sysoev6abfde62003-07-01 15:00:03 +0000245
Igor Sysoev630ad0c2004-04-16 05:14:16 +0000246 if (ngx_process == NGX_PROCESS_MASTER) {
Igor Sysoev924bd792004-10-11 15:07:03 +0000247 ngx_master_process_cycle(cycle);
Igor Sysoev630ad0c2004-04-16 05:14:16 +0000248
249 } else {
Igor Sysoev924bd792004-10-11 15:07:03 +0000250 ngx_single_process_cycle(cycle);
Igor Sysoev630ad0c2004-04-16 05:14:16 +0000251 }
Igor Sysoev3d58f8c2004-01-08 08:47:17 +0000252
253 return 0;
254}
255
256
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000257static ngx_int_t
258ngx_add_inherited_sockets(ngx_cycle_t *cycle)
Igor Sysoeva9030eb2004-01-06 16:49:34 +0000259{
Igor Sysoevc1571722005-03-19 12:38:37 +0000260 u_char *p, *v, *inherited;
261 ngx_int_t s;
262 ngx_listening_t *ls;
Igor Sysoeva9030eb2004-01-06 16:49:34 +0000263
Igor Sysoev10a543a2004-03-16 07:10:12 +0000264 inherited = (u_char *) getenv(NGINX_VAR);
Igor Sysoeva9030eb2004-01-06 16:49:34 +0000265
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000266 if (inherited == NULL) {
267 return NGX_OK;
Igor Sysoeva9030eb2004-01-06 16:49:34 +0000268 }
269
Igor Sysoev1b735832004-11-11 14:07:14 +0000270 ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0,
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000271 "using inherited sockets from \"%s\"", inherited);
272
Igor Sysoev1b735832004-11-11 14:07:14 +0000273 if (ngx_array_init(&cycle->listening, cycle->pool, 10,
274 sizeof(ngx_listening_t)) == NGX_ERROR)
275 {
276 return NGX_ERROR;
277 }
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000278
279 for (p = inherited, v = p; *p; p++) {
280 if (*p == ':' || *p == ';') {
281 s = ngx_atoi(v, p - v);
282 if (s == NGX_ERROR) {
283 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
284 "invalid socket number \"%s\" in "
Igor Sysoevc04deca2005-03-28 14:43:02 +0000285 NGINX_VAR " environment variable, "
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000286 "ignoring the rest of the variable", v);
287 break;
288 }
289
Igor Sysoev10a543a2004-03-16 07:10:12 +0000290 v = p + 1;
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000291
Igor Sysoevc1571722005-03-19 12:38:37 +0000292 ls = ngx_array_push(&cycle->listening);
293 if (ls == NULL) {
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000294 return NGX_ERROR;
Igor Sysoev10a543a2004-03-16 07:10:12 +0000295 }
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000296
Igor Sysoevc1571722005-03-19 12:38:37 +0000297 ls->fd = (ngx_socket_t) s;
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000298 }
299 }
300
301 ngx_inherited = 1;
302
303 return ngx_set_inherited_sockets(cycle);
Igor Sysoeva9030eb2004-01-06 16:49:34 +0000304}
305
306
Igor Sysoeva5362982004-03-04 07:04:55 +0000307ngx_pid_t ngx_exec_new_binary(ngx_cycle_t *cycle, char *const *argv)
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000308{
Igor Sysoev1b735832004-11-11 14:07:14 +0000309 char *env[3], *var;
310 u_char *p;
Igor Sysoev10a543a2004-03-16 07:10:12 +0000311 ngx_uint_t i;
Igor Sysoev80340f02004-01-13 21:33:59 +0000312 ngx_pid_t pid;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000313 ngx_exec_ctx_t ctx;
314 ngx_listening_t *ls;
315
316 ctx.path = argv[0];
317 ctx.name = "new binary process";
318 ctx.argv = argv;
319
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000320 var = ngx_alloc(sizeof(NGINX_VAR)
Igor Sysoevc1571722005-03-19 12:38:37 +0000321 + cycle->listening.nelts * (NGX_INT32_LEN + 1) + 2,
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000322 cycle->log);
323
Igor Sysoev1b735832004-11-11 14:07:14 +0000324 p = ngx_cpymem(var, NGINX_VAR "=", sizeof(NGINX_VAR));
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000325
326 ls = cycle->listening.elts;
327 for (i = 0; i < cycle->listening.nelts; i++) {
Igor Sysoev1b735832004-11-11 14:07:14 +0000328 p = ngx_sprintf(p, "%ud;", ls[i].fd);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000329 }
330
Igor Sysoev1b735832004-11-11 14:07:14 +0000331 *p = '\0';
332
Igor Sysoevfc5a10a2004-03-09 19:47:07 +0000333 ngx_log_debug1(NGX_LOG_DEBUG_CORE, cycle->log, 0, "inherited: %s", var);
334
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000335 env[0] = var;
Igor Sysoev924bd792004-10-11 15:07:03 +0000336
337#if (NGX_SETPROCTITLE_USES_ENV)
338
Igor Sysoev1b735832004-11-11 14:07:14 +0000339 /* allocate the spare 300 bytes for the new binary process title */
Igor Sysoev924bd792004-10-11 15:07:03 +0000340
341 env[1] = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"
342 "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"
343 "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"
344 "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"
345 "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";
346
347 env[2] = NULL;
348
349#else
350
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000351 env[1] = NULL;
Igor Sysoev924bd792004-10-11 15:07:03 +0000352
353#endif
354
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000355 ctx.envp = (char *const *) &env;
356
Igor Sysoev6a930452004-03-04 16:34:23 +0000357 pid = ngx_execute(cycle, &ctx);
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000358
359 ngx_free(var);
Igor Sysoev80340f02004-01-13 21:33:59 +0000360
361 return pid;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000362}
363
364
Igor Sysoev924bd792004-10-11 15:07:03 +0000365static ngx_int_t ngx_getopt(ngx_cycle_t *cycle, int argc, char *const *argv)
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000366{
367 ngx_int_t i;
368
Igor Sysoev924bd792004-10-11 15:07:03 +0000369 for (i = 1; i < argc; i++) {
370 if (argv[i][0] != '-') {
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000371 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
Igor Sysoev924bd792004-10-11 15:07:03 +0000372 "invalid option: \"%s\"", argv[i]);
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000373 return NGX_ERROR;
374 }
375
Igor Sysoev924bd792004-10-11 15:07:03 +0000376 switch (argv[i][1]) {
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000377
Igor Sysoev68df19d2004-04-15 15:34:36 +0000378 case 't':
379 ngx_test_config = 1;
380 break;
381
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000382 case 'c':
Igor Sysoev924bd792004-10-11 15:07:03 +0000383 if (argv[i + 1] == NULL) {
Igor Sysoevcccc5522004-04-14 20:34:05 +0000384 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
385 "the option: \"%s\" requires file name",
Igor Sysoev924bd792004-10-11 15:07:03 +0000386 argv[i]);
Igor Sysoevcccc5522004-04-14 20:34:05 +0000387 return NGX_ERROR;
388 }
389
Igor Sysoev924bd792004-10-11 15:07:03 +0000390 cycle->conf_file.data = (u_char *) argv[++i];
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000391 cycle->conf_file.len = ngx_strlen(cycle->conf_file.data);
392 break;
393
394 default:
395 ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
Igor Sysoev924bd792004-10-11 15:07:03 +0000396 "invalid option: \"%s\"", argv[i]);
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000397 return NGX_ERROR;
398 }
399 }
400
Igor Sysoevc7a2f682004-02-10 16:23:38 +0000401 if (cycle->conf_file.data == NULL) {
Igor Sysoev090849d2004-05-18 20:28:54 +0000402 cycle->conf_file.len = sizeof(NGX_CONF_PATH) - 1;
403 cycle->conf_file.data = (u_char *) NGX_CONF_PATH;
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000404 }
405
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000406 if (ngx_conf_full_name(cycle, &cycle->conf_file) == NGX_ERROR) {
407 return NGX_ERROR;
408 }
409
Igor Sysoeve9b2cb12004-02-09 20:47:18 +0000410 return NGX_OK;
411}
412
413
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000414static ngx_int_t
415ngx_save_argv(ngx_cycle_t *cycle, int argc, char *const *argv)
Igor Sysoev924bd792004-10-11 15:07:03 +0000416{
Igor Sysoevc1571722005-03-19 12:38:37 +0000417#if (NGX_FREEBSD)
418
419 ngx_os_argv = (char **) argv;
420 ngx_argc = argc;
421 ngx_argv = (char **) argv;
422
423#else
Igor Sysoev924bd792004-10-11 15:07:03 +0000424 size_t len;
425 ngx_int_t i;
426
427 ngx_os_argv = (char **) argv;
Igor Sysoev924bd792004-10-11 15:07:03 +0000428 ngx_argc = argc;
429
Igor Sysoevc1571722005-03-19 12:38:37 +0000430 ngx_argv = ngx_alloc((argc + 1) * sizeof(char *), cycle->log);
431 if (ngx_argv == NULL) {
Igor Sysoev924bd792004-10-11 15:07:03 +0000432 return NGX_ERROR;
433 }
434
435 for (i = 0; i < argc; i++) {
436 len = ngx_strlen(argv[i]) + 1;
437
Igor Sysoevc1571722005-03-19 12:38:37 +0000438 ngx_argv[i] = ngx_alloc(len, cycle->log);
439 if (ngx_argv[i] == NULL) {
Igor Sysoev924bd792004-10-11 15:07:03 +0000440 return NGX_ERROR;
441 }
442
Igor Sysoev4959ec42005-05-23 12:07:45 +0000443 (void) ngx_cpystrn((u_char *) ngx_argv[i], (u_char *) argv[i], len);
Igor Sysoev924bd792004-10-11 15:07:03 +0000444 }
445
446 ngx_argv[i] = NULL;
447
448#endif
449
450 return NGX_OK;
451}
452
453
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000454static void *
455ngx_core_module_create_conf(ngx_cycle_t *cycle)
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000456{
457 ngx_core_conf_t *ccf;
458
Igor Sysoevc1571722005-03-19 12:38:37 +0000459 ccf = ngx_pcalloc(cycle->pool, sizeof(ngx_core_conf_t));
460 if (ccf == NULL) {
Igor Sysoev43f13192004-04-12 16:38:09 +0000461 return NULL;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000462 }
Igor Sysoev02025fd2005-01-18 13:03:58 +0000463
464 /*
465 * set by pcalloc()
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000466 *
Igor Sysoev02025fd2005-01-18 13:03:58 +0000467 * ccf->pid = NULL;
468 * ccf->newpid = NULL;
469 * ccf->priority = 0;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000470 */
Igor Sysoev02025fd2005-01-18 13:03:58 +0000471
Igor Sysoev03420a62004-01-20 20:40:08 +0000472 ccf->daemon = NGX_CONF_UNSET;
473 ccf->master = NGX_CONF_UNSET;
Igor Sysoeva741f8d2004-03-30 20:31:58 +0000474 ccf->worker_processes = NGX_CONF_UNSET;
Igor Sysoev1b735832004-11-11 14:07:14 +0000475 ccf->debug_points = NGX_CONF_UNSET;
Igor Sysoev4959ec42005-05-23 12:07:45 +0000476 ccf->user = (ngx_uid_t) NGX_CONF_UNSET_UINT;
477 ccf->group = (ngx_gid_t) NGX_CONF_UNSET_UINT;
Igor Sysoev1b735832004-11-11 14:07:14 +0000478#if (NGX_THREADS)
479 ccf->worker_threads = NGX_CONF_UNSET;
480 ccf->thread_stack_size = NGX_CONF_UNSET_SIZE;
481#endif
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000482
Igor Sysoev43f13192004-04-12 16:38:09 +0000483 return ccf;
484}
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000485
Igor Sysoev43f13192004-04-12 16:38:09 +0000486
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000487static char *
488ngx_core_module_init_conf(ngx_cycle_t *cycle, void *conf)
Igor Sysoev43f13192004-04-12 16:38:09 +0000489{
490 ngx_core_conf_t *ccf = conf;
491
Igor Sysoev1b735832004-11-11 14:07:14 +0000492#if !(NGX_WIN32)
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000493 struct passwd *pwd;
494 struct group *grp;
495#endif
496
Igor Sysoev43f13192004-04-12 16:38:09 +0000497 ngx_conf_init_value(ccf->daemon, 1);
498 ngx_conf_init_value(ccf->master, 1);
499 ngx_conf_init_value(ccf->worker_processes, 1);
Igor Sysoev1b735832004-11-11 14:07:14 +0000500 ngx_conf_init_value(ccf->debug_points, 0);
Igor Sysoev43f13192004-04-12 16:38:09 +0000501
Igor Sysoev32fcd5c2004-07-05 06:55:54 +0000502#if (NGX_THREADS)
503 ngx_conf_init_value(ccf->worker_threads, 0);
504 ngx_threads_n = ccf->worker_threads;
505 ngx_conf_init_size_value(ccf->thread_stack_size, 2 * 1024 * 1024);
506#endif
507
Igor Sysoev1b735832004-11-11 14:07:14 +0000508#if !(NGX_WIN32)
Igor Sysoev43f13192004-04-12 16:38:09 +0000509
Igor Sysoev4959ec42005-05-23 12:07:45 +0000510 if (ccf->user == (uid_t) NGX_CONF_UNSET_UINT && geteuid() == 0) {
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000511
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000512 pwd = getpwnam(NGX_USER);
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000513 if (pwd == NULL) {
514 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000515 "getpwnam(\"" NGX_USER "\") failed");
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000516 return NGX_CONF_ERROR;
517 }
518
Igor Sysoev42b12b32004-12-02 18:40:46 +0000519 ccf->username = NGX_USER;
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000520 ccf->user = pwd->pw_uid;
521
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000522 grp = getgrnam(NGX_GROUP);
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000523 if (grp == NULL) {
524 ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000525 "getgrnam(\"" NGX_GROUP "\") failed");
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000526 return NGX_CONF_ERROR;
527 }
528
529 ccf->group = grp->gr_gid;
530 }
Igor Sysoev43f13192004-04-12 16:38:09 +0000531
532 if (ccf->pid.len == 0) {
Igor Sysoev090849d2004-05-18 20:28:54 +0000533 ccf->pid.len = sizeof(NGX_PID_PATH) - 1;
Igor Sysoev924bd792004-10-11 15:07:03 +0000534 ccf->pid.data = (u_char *) NGX_PID_PATH;
Igor Sysoev43f13192004-04-12 16:38:09 +0000535 }
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000536
537 if (ngx_conf_full_name(cycle, &ccf->pid) == NGX_ERROR) {
538 return NGX_CONF_ERROR;
539 }
540
541 ccf->newpid.len = ccf->pid.len + sizeof(NGX_NEWPID_EXT);
542
Igor Sysoevc1571722005-03-19 12:38:37 +0000543 ccf->newpid.data = ngx_palloc(cycle->pool, ccf->newpid.len);
544 if (ccf->newpid.data == NULL) {
Igor Sysoev6d2a14a2004-09-27 16:03:21 +0000545 return NGX_CONF_ERROR;
546 }
547
548 ngx_memcpy(ngx_cpymem(ccf->newpid.data, ccf->pid.data, ccf->pid.len),
549 NGX_NEWPID_EXT, sizeof(NGX_NEWPID_EXT));
550
Igor Sysoev43f13192004-04-12 16:38:09 +0000551#endif
552
553 return NGX_CONF_OK;
Igor Sysoev3c3ca172004-01-05 20:55:48 +0000554}
Igor Sysoev03420a62004-01-20 20:40:08 +0000555
556
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000557static char *
558ngx_set_user(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
Igor Sysoev03420a62004-01-20 20:40:08 +0000559{
Igor Sysoev1b735832004-11-11 14:07:14 +0000560#if (NGX_WIN32)
Igor Sysoeva5362982004-03-04 07:04:55 +0000561
562 ngx_conf_log_error(NGX_LOG_WARN, cf, 0,
563 "\"user\" is not supported, ignored");
564
565 return NGX_CONF_OK;
566
567#else
568
Igor Sysoev43f13192004-04-12 16:38:09 +0000569 ngx_core_conf_t *ccf = conf;
570
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000571 char *group;
Igor Sysoev03420a62004-01-20 20:40:08 +0000572 struct passwd *pwd;
573 struct group *grp;
574 ngx_str_t *value;
Igor Sysoev03420a62004-01-20 20:40:08 +0000575
Igor Sysoev4959ec42005-05-23 12:07:45 +0000576 if (ccf->user != (uid_t) NGX_CONF_UNSET_UINT) {
Igor Sysoev03420a62004-01-20 20:40:08 +0000577 return "is duplicate";
578 }
579
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000580 if (geteuid() != 0) {
581 ngx_conf_log_error(NGX_LOG_WARN, cf, 0,
582 "the \"user\" directive makes sense only "
583 "if the master process runs "
584 "with super-user privileges, ignored");
585 return NGX_CONF_OK;
586 }
587
Igor Sysoev03420a62004-01-20 20:40:08 +0000588 value = (ngx_str_t *) cf->args->elts;
589
Igor Sysoev42b12b32004-12-02 18:40:46 +0000590 ccf->username = (char *) value[1].data;
591
Igor Sysoev10a543a2004-03-16 07:10:12 +0000592 pwd = getpwnam((const char *) value[1].data);
Igor Sysoev03420a62004-01-20 20:40:08 +0000593 if (pwd == NULL) {
594 ngx_conf_log_error(NGX_LOG_EMERG, cf, ngx_errno,
Igor Sysoev924bd792004-10-11 15:07:03 +0000595 "getpwnam(\"%s\") failed", value[1].data);
Igor Sysoev03420a62004-01-20 20:40:08 +0000596 return NGX_CONF_ERROR;
597 }
598
599 ccf->user = pwd->pw_uid;
600
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000601 group = (char *) ((cf->args->nelts == 2) ? value[1].data : value[2].data);
Igor Sysoev03420a62004-01-20 20:40:08 +0000602
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000603 grp = getgrnam(group);
Igor Sysoev03420a62004-01-20 20:40:08 +0000604 if (grp == NULL) {
605 ngx_conf_log_error(NGX_LOG_EMERG, cf, ngx_errno,
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000606 "getgrnam(\"%s\") failed", group);
Igor Sysoev03420a62004-01-20 20:40:08 +0000607 return NGX_CONF_ERROR;
608 }
609
610 ccf->group = grp->gr_gid;
611
612 return NGX_CONF_OK;
Igor Sysoeva5362982004-03-04 07:04:55 +0000613
614#endif
Igor Sysoev03420a62004-01-20 20:40:08 +0000615}
Igor Sysoev42b12b32004-12-02 18:40:46 +0000616
617
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000618static char *
619ngx_set_priority(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
Igor Sysoev42b12b32004-12-02 18:40:46 +0000620{
621 ngx_core_conf_t *ccf = conf;
622
623 ngx_str_t *value;
624 ngx_uint_t n, minus;
625
626 if (ccf->priority != 0) {
627 return "is duplicate";
628 }
629
630 value = cf->args->elts;
631
632 if (value[1].data[0] == '-') {
633 n = 1;
634 minus = 1;
635
636 } else if (value[1].data[0] == '+') {
637 n = 1;
638 minus = 0;
639
640 } else {
641 n = 0;
642 minus = 0;
643 }
644
645 ccf->priority = ngx_atoi(&value[1].data[n], value[1].len - n);
646 if (ccf->priority == NGX_ERROR) {
647 return "invalid number";
648 }
649
650 if (minus) {
651 ccf->priority = -ccf->priority;
652 }
653
654 return NGX_CONF_OK;
655}