| |
| /* |
| * Copyright (C) Igor Sysoev |
| * Copyright (C) Nginx, Inc. |
| */ |
| |
| |
| #include <ngx_config.h> |
| #include <ngx_core.h> |
| #include <nginx.h> |
| |
| |
| static void ngx_show_version_info(void); |
| static ngx_int_t ngx_add_inherited_sockets(ngx_cycle_t *cycle); |
| static void ngx_cleanup_environment(void *data); |
| static ngx_int_t ngx_get_options(int argc, char *const *argv); |
| static ngx_int_t ngx_process_options(ngx_cycle_t *cycle); |
| static ngx_int_t ngx_save_argv(ngx_cycle_t *cycle, int argc, char *const *argv); |
| static void *ngx_core_module_create_conf(ngx_cycle_t *cycle); |
| static char *ngx_core_module_init_conf(ngx_cycle_t *cycle, void *conf); |
| static char *ngx_set_user(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); |
| static char *ngx_set_env(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); |
| static char *ngx_set_priority(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); |
| static char *ngx_set_cpu_affinity(ngx_conf_t *cf, ngx_command_t *cmd, |
| void *conf); |
| static char *ngx_set_worker_processes(ngx_conf_t *cf, ngx_command_t *cmd, |
| void *conf); |
| static char *ngx_load_module(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); |
| #if (NGX_HAVE_DLOPEN) |
| static void ngx_unload_module(void *data); |
| #endif |
| |
| |
| static ngx_conf_enum_t ngx_debug_points[] = { |
| { ngx_string("stop"), NGX_DEBUG_POINTS_STOP }, |
| { ngx_string("abort"), NGX_DEBUG_POINTS_ABORT }, |
| { ngx_null_string, 0 } |
| }; |
| |
| |
| static ngx_command_t ngx_core_commands[] = { |
| |
| { ngx_string("daemon"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_FLAG, |
| ngx_conf_set_flag_slot, |
| 0, |
| offsetof(ngx_core_conf_t, daemon), |
| NULL }, |
| |
| { ngx_string("master_process"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_FLAG, |
| ngx_conf_set_flag_slot, |
| 0, |
| offsetof(ngx_core_conf_t, master), |
| NULL }, |
| |
| { ngx_string("timer_resolution"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_msec_slot, |
| 0, |
| offsetof(ngx_core_conf_t, timer_resolution), |
| NULL }, |
| |
| { ngx_string("pid"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_str_slot, |
| 0, |
| offsetof(ngx_core_conf_t, pid), |
| NULL }, |
| |
| { ngx_string("lock_file"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_str_slot, |
| 0, |
| offsetof(ngx_core_conf_t, lock_file), |
| NULL }, |
| |
| { ngx_string("worker_processes"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_set_worker_processes, |
| 0, |
| 0, |
| NULL }, |
| |
| { ngx_string("debug_points"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_enum_slot, |
| 0, |
| offsetof(ngx_core_conf_t, debug_points), |
| &ngx_debug_points }, |
| |
| { ngx_string("user"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE12, |
| ngx_set_user, |
| 0, |
| 0, |
| NULL }, |
| |
| { ngx_string("worker_priority"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_set_priority, |
| 0, |
| 0, |
| NULL }, |
| |
| { ngx_string("worker_cpu_affinity"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_1MORE, |
| ngx_set_cpu_affinity, |
| 0, |
| 0, |
| NULL }, |
| |
| { ngx_string("worker_rlimit_nofile"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_num_slot, |
| 0, |
| offsetof(ngx_core_conf_t, rlimit_nofile), |
| NULL }, |
| |
| { ngx_string("worker_rlimit_core"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_off_slot, |
| 0, |
| offsetof(ngx_core_conf_t, rlimit_core), |
| NULL }, |
| |
| { ngx_string("worker_shutdown_timeout"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_msec_slot, |
| 0, |
| offsetof(ngx_core_conf_t, shutdown_timeout), |
| NULL }, |
| |
| { ngx_string("working_directory"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_conf_set_str_slot, |
| 0, |
| offsetof(ngx_core_conf_t, working_directory), |
| NULL }, |
| |
| { ngx_string("env"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_set_env, |
| 0, |
| 0, |
| NULL }, |
| |
| { ngx_string("load_module"), |
| NGX_MAIN_CONF|NGX_DIRECT_CONF|NGX_CONF_TAKE1, |
| ngx_load_module, |
| 0, |
| 0, |
| NULL }, |
| |
| ngx_null_command |
| }; |
| |
| |
| static ngx_core_module_t ngx_core_module_ctx = { |
| ngx_string("core"), |
| ngx_core_module_create_conf, |
| ngx_core_module_init_conf |
| }; |
| |
| |
| ngx_module_t ngx_core_module = { |
| NGX_MODULE_V1, |
| &ngx_core_module_ctx, /* module context */ |
| ngx_core_commands, /* module directives */ |
| NGX_CORE_MODULE, /* module type */ |
| NULL, /* init master */ |
| NULL, /* init module */ |
| NULL, /* init process */ |
| NULL, /* init thread */ |
| NULL, /* exit thread */ |
| NULL, /* exit process */ |
| NULL, /* exit master */ |
| NGX_MODULE_V1_PADDING |
| }; |
| |
| |
| static ngx_uint_t ngx_show_help; |
| static ngx_uint_t ngx_show_version; |
| static ngx_uint_t ngx_show_configure; |
| static u_char *ngx_prefix; |
| static u_char *ngx_error_log; |
| static u_char *ngx_conf_file; |
| static u_char *ngx_conf_params; |
| static char *ngx_signal; |
| |
| |
| static char **ngx_os_environ; |
| |
| |
| int ngx_cdecl |
| main(int argc, char *const *argv) |
| { |
| ngx_buf_t *b; |
| ngx_log_t *log; |
| ngx_uint_t i; |
| ngx_cycle_t *cycle, init_cycle; |
| ngx_conf_dump_t *cd; |
| ngx_core_conf_t *ccf; |
| |
| ngx_debug_init(); |
| |
| if (ngx_strerror_init() != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_get_options(argc, argv) != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_show_version) { |
| ngx_show_version_info(); |
| |
| if (!ngx_test_config) { |
| return 0; |
| } |
| } |
| |
| /* TODO */ ngx_max_sockets = -1; |
| |
| ngx_time_init(); |
| |
| #if (NGX_PCRE) |
| ngx_regex_init(); |
| #endif |
| |
| ngx_pid = ngx_getpid(); |
| ngx_parent = ngx_getppid(); |
| |
| log = ngx_log_init(ngx_prefix, ngx_error_log); |
| if (log == NULL) { |
| return 1; |
| } |
| |
| /* STUB */ |
| #if (NGX_OPENSSL) |
| ngx_ssl_init(log); |
| #endif |
| |
| /* |
| * init_cycle->log is required for signal handlers and |
| * ngx_process_options() |
| */ |
| |
| ngx_memzero(&init_cycle, sizeof(ngx_cycle_t)); |
| init_cycle.log = log; |
| ngx_cycle = &init_cycle; |
| |
| init_cycle.pool = ngx_create_pool(1024, log); |
| if (init_cycle.pool == NULL) { |
| return 1; |
| } |
| |
| if (ngx_save_argv(&init_cycle, argc, argv) != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_process_options(&init_cycle) != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_os_init(log) != NGX_OK) { |
| return 1; |
| } |
| |
| /* |
| * ngx_crc32_table_init() requires ngx_cacheline_size set in ngx_os_init() |
| */ |
| |
| if (ngx_crc32_table_init() != NGX_OK) { |
| return 1; |
| } |
| |
| /* |
| * ngx_slab_sizes_init() requires ngx_pagesize set in ngx_os_init() |
| */ |
| |
| ngx_slab_sizes_init(); |
| |
| if (ngx_add_inherited_sockets(&init_cycle) != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_preinit_modules() != NGX_OK) { |
| return 1; |
| } |
| |
| cycle = ngx_init_cycle(&init_cycle); |
| if (cycle == NULL) { |
| if (ngx_test_config) { |
| ngx_log_stderr(0, "configuration file %s test failed", |
| init_cycle.conf_file.data); |
| } |
| |
| return 1; |
| } |
| |
| if (ngx_test_config) { |
| if (!ngx_quiet_mode) { |
| ngx_log_stderr(0, "configuration file %s test is successful", |
| cycle->conf_file.data); |
| } |
| |
| if (ngx_dump_config) { |
| cd = cycle->config_dump.elts; |
| |
| for (i = 0; i < cycle->config_dump.nelts; i++) { |
| |
| ngx_write_stdout("# configuration file "); |
| (void) ngx_write_fd(ngx_stdout, cd[i].name.data, |
| cd[i].name.len); |
| ngx_write_stdout(":" NGX_LINEFEED); |
| |
| b = cd[i].buffer; |
| |
| (void) ngx_write_fd(ngx_stdout, b->pos, b->last - b->pos); |
| ngx_write_stdout(NGX_LINEFEED); |
| } |
| } |
| |
| return 0; |
| } |
| |
| if (ngx_signal) { |
| return ngx_signal_process(cycle, ngx_signal); |
| } |
| |
| ngx_os_status(cycle->log); |
| |
| ngx_cycle = cycle; |
| |
| ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module); |
| |
| if (ccf->master && ngx_process == NGX_PROCESS_SINGLE) { |
| ngx_process = NGX_PROCESS_MASTER; |
| } |
| |
| #if !(NGX_WIN32) |
| |
| if (ngx_init_signals(cycle->log) != NGX_OK) { |
| return 1; |
| } |
| |
| if (!ngx_inherited && ccf->daemon) { |
| if (ngx_daemon(cycle->log) != NGX_OK) { |
| return 1; |
| } |
| |
| ngx_daemonized = 1; |
| } |
| |
| if (ngx_inherited) { |
| ngx_daemonized = 1; |
| } |
| |
| #endif |
| |
| if (ngx_create_pidfile(&ccf->pid, cycle->log) != NGX_OK) { |
| return 1; |
| } |
| |
| if (ngx_log_redirect_stderr(cycle) != NGX_OK) { |
| return 1; |
| } |
| |
| if (log->file->fd != ngx_stderr) { |
| if (ngx_close_file(log->file->fd) == NGX_FILE_ERROR) { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| ngx_close_file_n " built-in log failed"); |
| } |
| } |
| |
| ngx_use_stderr = 0; |
| |
| if (ngx_process == NGX_PROCESS_SINGLE) { |
| ngx_single_process_cycle(cycle); |
| |
| } else { |
| ngx_master_process_cycle(cycle); |
| } |
| |
| return 0; |
| } |
| |
| |
| static void |
| ngx_show_version_info(void) |
| { |
| ngx_write_stderr("nginx version: " NGINX_VER_BUILD NGX_LINEFEED); |
| |
| if (ngx_show_help) { |
| ngx_write_stderr( |
| "Usage: nginx [-?hvVtTq] [-s signal] [-p prefix]" NGX_LINEFEED |
| " [-e filename] [-c filename] [-g directives]" |
| NGX_LINEFEED NGX_LINEFEED |
| "Options:" NGX_LINEFEED |
| " -?,-h : this help" NGX_LINEFEED |
| " -v : show version and exit" NGX_LINEFEED |
| " -V : show version and configure options then exit" |
| NGX_LINEFEED |
| " -t : test configuration and exit" NGX_LINEFEED |
| " -T : test configuration, dump it and exit" |
| NGX_LINEFEED |
| " -q : suppress non-error messages " |
| "during configuration testing" NGX_LINEFEED |
| " -s signal : send signal to a master process: " |
| "stop, quit, reopen, reload" NGX_LINEFEED |
| #ifdef NGX_PREFIX |
| " -p prefix : set prefix path (default: " NGX_PREFIX ")" |
| NGX_LINEFEED |
| #else |
| " -p prefix : set prefix path (default: NONE)" NGX_LINEFEED |
| #endif |
| " -e filename : set error log file (default: " |
| #ifdef NGX_ERROR_LOG_STDERR |
| "stderr)" NGX_LINEFEED |
| #else |
| NGX_ERROR_LOG_PATH ")" NGX_LINEFEED |
| #endif |
| " -c filename : set configuration file (default: " NGX_CONF_PATH |
| ")" NGX_LINEFEED |
| " -g directives : set global directives out of configuration " |
| "file" NGX_LINEFEED NGX_LINEFEED |
| ); |
| } |
| |
| if (ngx_show_configure) { |
| |
| #ifdef NGX_COMPILER |
| ngx_write_stderr("built by " NGX_COMPILER NGX_LINEFEED); |
| #endif |
| |
| #if (NGX_SSL) |
| if (ngx_strcmp(ngx_ssl_version(), OPENSSL_VERSION_TEXT) == 0) { |
| ngx_write_stderr("built with " OPENSSL_VERSION_TEXT NGX_LINEFEED); |
| } else { |
| ngx_write_stderr("built with " OPENSSL_VERSION_TEXT |
| " (running with "); |
| ngx_write_stderr((char *) (uintptr_t) ngx_ssl_version()); |
| ngx_write_stderr(")" NGX_LINEFEED); |
| } |
| #ifdef SSL_CTRL_SET_TLSEXT_HOSTNAME |
| ngx_write_stderr("TLS SNI support enabled" NGX_LINEFEED); |
| #else |
| ngx_write_stderr("TLS SNI support disabled" NGX_LINEFEED); |
| #endif |
| #endif |
| |
| ngx_write_stderr("configure arguments:" NGX_CONFIGURE NGX_LINEFEED); |
| } |
| } |
| |
| |
| static ngx_int_t |
| ngx_add_inherited_sockets(ngx_cycle_t *cycle) |
| { |
| u_char *p, *v, *inherited; |
| ngx_int_t s; |
| ngx_listening_t *ls; |
| |
| inherited = (u_char *) getenv(NGINX_VAR); |
| |
| if (inherited == NULL) { |
| return NGX_OK; |
| } |
| |
| ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, |
| "using inherited sockets from \"%s\"", inherited); |
| |
| if (ngx_array_init(&cycle->listening, cycle->pool, 10, |
| sizeof(ngx_listening_t)) |
| != NGX_OK) |
| { |
| return NGX_ERROR; |
| } |
| |
| for (p = inherited, v = p; *p; p++) { |
| if (*p == ':' || *p == ';') { |
| s = ngx_atoi(v, p - v); |
| if (s == NGX_ERROR) { |
| ngx_log_error(NGX_LOG_EMERG, cycle->log, 0, |
| "invalid socket number \"%s\" in " NGINX_VAR |
| " environment variable, ignoring the rest" |
| " of the variable", v); |
| break; |
| } |
| |
| v = p + 1; |
| |
| ls = ngx_array_push(&cycle->listening); |
| if (ls == NULL) { |
| return NGX_ERROR; |
| } |
| |
| ngx_memzero(ls, sizeof(ngx_listening_t)); |
| |
| ls->fd = (ngx_socket_t) s; |
| ls->inherited = 1; |
| } |
| } |
| |
| if (v != p) { |
| ngx_log_error(NGX_LOG_EMERG, cycle->log, 0, |
| "invalid socket number \"%s\" in " NGINX_VAR |
| " environment variable, ignoring", v); |
| } |
| |
| ngx_inherited = 1; |
| |
| return ngx_set_inherited_sockets(cycle); |
| } |
| |
| |
| char ** |
| ngx_set_environment(ngx_cycle_t *cycle, ngx_uint_t *last) |
| { |
| char **p, **env; |
| ngx_str_t *var; |
| ngx_uint_t i, n; |
| ngx_core_conf_t *ccf; |
| ngx_pool_cleanup_t *cln; |
| |
| ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module); |
| |
| if (last == NULL && ccf->environment) { |
| return ccf->environment; |
| } |
| |
| var = ccf->env.elts; |
| |
| for (i = 0; i < ccf->env.nelts; i++) { |
| if (ngx_strcmp(var[i].data, "TZ") == 0 |
| || ngx_strncmp(var[i].data, "TZ=", 3) == 0) |
| { |
| goto tz_found; |
| } |
| } |
| |
| var = ngx_array_push(&ccf->env); |
| if (var == NULL) { |
| return NULL; |
| } |
| |
| var->len = 2; |
| var->data = (u_char *) "TZ"; |
| |
| var = ccf->env.elts; |
| |
| tz_found: |
| |
| n = 0; |
| |
| for (i = 0; i < ccf->env.nelts; i++) { |
| |
| if (var[i].data[var[i].len] == '=') { |
| n++; |
| continue; |
| } |
| |
| for (p = ngx_os_environ; *p; p++) { |
| |
| if (ngx_strncmp(*p, var[i].data, var[i].len) == 0 |
| && (*p)[var[i].len] == '=') |
| { |
| n++; |
| break; |
| } |
| } |
| } |
| |
| if (last) { |
| env = ngx_alloc((*last + n + 1) * sizeof(char *), cycle->log); |
| if (env == NULL) { |
| return NULL; |
| } |
| |
| *last = n; |
| |
| } else { |
| cln = ngx_pool_cleanup_add(cycle->pool, 0); |
| if (cln == NULL) { |
| return NULL; |
| } |
| |
| env = ngx_alloc((n + 1) * sizeof(char *), cycle->log); |
| if (env == NULL) { |
| return NULL; |
| } |
| |
| cln->handler = ngx_cleanup_environment; |
| cln->data = env; |
| } |
| |
| n = 0; |
| |
| for (i = 0; i < ccf->env.nelts; i++) { |
| |
| if (var[i].data[var[i].len] == '=') { |
| env[n++] = (char *) var[i].data; |
| continue; |
| } |
| |
| for (p = ngx_os_environ; *p; p++) { |
| |
| if (ngx_strncmp(*p, var[i].data, var[i].len) == 0 |
| && (*p)[var[i].len] == '=') |
| { |
| env[n++] = *p; |
| break; |
| } |
| } |
| } |
| |
| env[n] = NULL; |
| |
| if (last == NULL) { |
| ccf->environment = env; |
| environ = env; |
| } |
| |
| return env; |
| } |
| |
| |
| static void |
| ngx_cleanup_environment(void *data) |
| { |
| char **env = data; |
| |
| if (environ == env) { |
| |
| /* |
| * if the environment is still used, as it happens on exit, |
| * the only option is to leak it |
| */ |
| |
| return; |
| } |
| |
| ngx_free(env); |
| } |
| |
| |
| ngx_pid_t |
| ngx_exec_new_binary(ngx_cycle_t *cycle, char *const *argv) |
| { |
| char **env, *var; |
| u_char *p; |
| ngx_uint_t i, n; |
| ngx_pid_t pid; |
| ngx_exec_ctx_t ctx; |
| ngx_core_conf_t *ccf; |
| ngx_listening_t *ls; |
| |
| ngx_memzero(&ctx, sizeof(ngx_exec_ctx_t)); |
| |
| ctx.path = argv[0]; |
| ctx.name = "new binary process"; |
| ctx.argv = argv; |
| |
| n = 2; |
| env = ngx_set_environment(cycle, &n); |
| if (env == NULL) { |
| return NGX_INVALID_PID; |
| } |
| |
| var = ngx_alloc(sizeof(NGINX_VAR) |
| + cycle->listening.nelts * (NGX_INT32_LEN + 1) + 2, |
| cycle->log); |
| if (var == NULL) { |
| ngx_free(env); |
| return NGX_INVALID_PID; |
| } |
| |
| p = ngx_cpymem(var, NGINX_VAR "=", sizeof(NGINX_VAR)); |
| |
| ls = cycle->listening.elts; |
| for (i = 0; i < cycle->listening.nelts; i++) { |
| p = ngx_sprintf(p, "%ud;", ls[i].fd); |
| } |
| |
| *p = '\0'; |
| |
| env[n++] = var; |
| |
| #if (NGX_SETPROCTITLE_USES_ENV) |
| |
| /* allocate the spare 300 bytes for the new binary process title */ |
| |
| env[n++] = "SPARE=XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX" |
| "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX" |
| "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX" |
| "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX" |
| "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"; |
| |
| #endif |
| |
| env[n] = NULL; |
| |
| #if (NGX_DEBUG) |
| { |
| char **e; |
| for (e = env; *e; e++) { |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, cycle->log, 0, "env: %s", *e); |
| } |
| } |
| #endif |
| |
| ctx.envp = (char *const *) env; |
| |
| ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module); |
| |
| if (ngx_rename_file(ccf->pid.data, ccf->oldpid.data) == NGX_FILE_ERROR) { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| ngx_rename_file_n " %s to %s failed " |
| "before executing new binary process \"%s\"", |
| ccf->pid.data, ccf->oldpid.data, argv[0]); |
| |
| ngx_free(env); |
| ngx_free(var); |
| |
| return NGX_INVALID_PID; |
| } |
| |
| pid = ngx_execute(cycle, &ctx); |
| |
| if (pid == NGX_INVALID_PID) { |
| if (ngx_rename_file(ccf->oldpid.data, ccf->pid.data) |
| == NGX_FILE_ERROR) |
| { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| ngx_rename_file_n " %s back to %s failed after " |
| "an attempt to execute new binary process \"%s\"", |
| ccf->oldpid.data, ccf->pid.data, argv[0]); |
| } |
| } |
| |
| ngx_free(env); |
| ngx_free(var); |
| |
| return pid; |
| } |
| |
| |
| static ngx_int_t |
| ngx_get_options(int argc, char *const *argv) |
| { |
| u_char *p; |
| ngx_int_t i; |
| |
| for (i = 1; i < argc; i++) { |
| |
| p = (u_char *) argv[i]; |
| |
| if (*p++ != '-') { |
| ngx_log_stderr(0, "invalid option: \"%s\"", argv[i]); |
| return NGX_ERROR; |
| } |
| |
| while (*p) { |
| |
| switch (*p++) { |
| |
| case '?': |
| case 'h': |
| ngx_show_version = 1; |
| ngx_show_help = 1; |
| break; |
| |
| case 'v': |
| ngx_show_version = 1; |
| break; |
| |
| case 'V': |
| ngx_show_version = 1; |
| ngx_show_configure = 1; |
| break; |
| |
| case 't': |
| ngx_test_config = 1; |
| break; |
| |
| case 'T': |
| ngx_test_config = 1; |
| ngx_dump_config = 1; |
| break; |
| |
| case 'q': |
| ngx_quiet_mode = 1; |
| break; |
| |
| case 'p': |
| if (*p) { |
| ngx_prefix = p; |
| goto next; |
| } |
| |
| if (argv[++i]) { |
| ngx_prefix = (u_char *) argv[i]; |
| goto next; |
| } |
| |
| ngx_log_stderr(0, "option \"-p\" requires directory name"); |
| return NGX_ERROR; |
| |
| case 'e': |
| if (*p) { |
| ngx_error_log = p; |
| |
| } else if (argv[++i]) { |
| ngx_error_log = (u_char *) argv[i]; |
| |
| } else { |
| ngx_log_stderr(0, "option \"-e\" requires file name"); |
| return NGX_ERROR; |
| } |
| |
| if (ngx_strcmp(ngx_error_log, "stderr") == 0) { |
| ngx_error_log = (u_char *) ""; |
| } |
| |
| goto next; |
| |
| case 'c': |
| if (*p) { |
| ngx_conf_file = p; |
| goto next; |
| } |
| |
| if (argv[++i]) { |
| ngx_conf_file = (u_char *) argv[i]; |
| goto next; |
| } |
| |
| ngx_log_stderr(0, "option \"-c\" requires file name"); |
| return NGX_ERROR; |
| |
| case 'g': |
| if (*p) { |
| ngx_conf_params = p; |
| goto next; |
| } |
| |
| if (argv[++i]) { |
| ngx_conf_params = (u_char *) argv[i]; |
| goto next; |
| } |
| |
| ngx_log_stderr(0, "option \"-g\" requires parameter"); |
| return NGX_ERROR; |
| |
| case 's': |
| if (*p) { |
| ngx_signal = (char *) p; |
| |
| } else if (argv[++i]) { |
| ngx_signal = argv[i]; |
| |
| } else { |
| ngx_log_stderr(0, "option \"-s\" requires parameter"); |
| return NGX_ERROR; |
| } |
| |
| if (ngx_strcmp(ngx_signal, "stop") == 0 |
| || ngx_strcmp(ngx_signal, "quit") == 0 |
| || ngx_strcmp(ngx_signal, "reopen") == 0 |
| || ngx_strcmp(ngx_signal, "reload") == 0) |
| { |
| ngx_process = NGX_PROCESS_SIGNALLER; |
| goto next; |
| } |
| |
| ngx_log_stderr(0, "invalid option: \"-s %s\"", ngx_signal); |
| return NGX_ERROR; |
| |
| default: |
| ngx_log_stderr(0, "invalid option: \"%c\"", *(p - 1)); |
| return NGX_ERROR; |
| } |
| } |
| |
| next: |
| |
| continue; |
| } |
| |
| return NGX_OK; |
| } |
| |
| |
| static ngx_int_t |
| ngx_save_argv(ngx_cycle_t *cycle, int argc, char *const *argv) |
| { |
| #if (NGX_FREEBSD) |
| |
| ngx_os_argv = (char **) argv; |
| ngx_argc = argc; |
| ngx_argv = (char **) argv; |
| |
| #else |
| size_t len; |
| ngx_int_t i; |
| |
| ngx_os_argv = (char **) argv; |
| ngx_argc = argc; |
| |
| ngx_argv = ngx_alloc((argc + 1) * sizeof(char *), cycle->log); |
| if (ngx_argv == NULL) { |
| return NGX_ERROR; |
| } |
| |
| for (i = 0; i < argc; i++) { |
| len = ngx_strlen(argv[i]) + 1; |
| |
| ngx_argv[i] = ngx_alloc(len, cycle->log); |
| if (ngx_argv[i] == NULL) { |
| return NGX_ERROR; |
| } |
| |
| (void) ngx_cpystrn((u_char *) ngx_argv[i], (u_char *) argv[i], len); |
| } |
| |
| ngx_argv[i] = NULL; |
| |
| #endif |
| |
| ngx_os_environ = environ; |
| |
| return NGX_OK; |
| } |
| |
| |
| static ngx_int_t |
| ngx_process_options(ngx_cycle_t *cycle) |
| { |
| u_char *p; |
| size_t len; |
| |
| if (ngx_prefix) { |
| len = ngx_strlen(ngx_prefix); |
| p = ngx_prefix; |
| |
| if (len && !ngx_path_separator(p[len - 1])) { |
| p = ngx_pnalloc(cycle->pool, len + 1); |
| if (p == NULL) { |
| return NGX_ERROR; |
| } |
| |
| ngx_memcpy(p, ngx_prefix, len); |
| p[len++] = '/'; |
| } |
| |
| cycle->conf_prefix.len = len; |
| cycle->conf_prefix.data = p; |
| cycle->prefix.len = len; |
| cycle->prefix.data = p; |
| |
| } else { |
| |
| #ifndef NGX_PREFIX |
| |
| p = ngx_pnalloc(cycle->pool, NGX_MAX_PATH); |
| if (p == NULL) { |
| return NGX_ERROR; |
| } |
| |
| if (ngx_getcwd(p, NGX_MAX_PATH) == 0) { |
| ngx_log_stderr(ngx_errno, "[emerg]: " ngx_getcwd_n " failed"); |
| return NGX_ERROR; |
| } |
| |
| len = ngx_strlen(p); |
| |
| p[len++] = '/'; |
| |
| cycle->conf_prefix.len = len; |
| cycle->conf_prefix.data = p; |
| cycle->prefix.len = len; |
| cycle->prefix.data = p; |
| |
| #else |
| |
| #ifdef NGX_CONF_PREFIX |
| ngx_str_set(&cycle->conf_prefix, NGX_CONF_PREFIX); |
| #else |
| ngx_str_set(&cycle->conf_prefix, NGX_PREFIX); |
| #endif |
| ngx_str_set(&cycle->prefix, NGX_PREFIX); |
| |
| #endif |
| } |
| |
| if (ngx_conf_file) { |
| cycle->conf_file.len = ngx_strlen(ngx_conf_file); |
| cycle->conf_file.data = ngx_conf_file; |
| |
| } else { |
| ngx_str_set(&cycle->conf_file, NGX_CONF_PATH); |
| } |
| |
| if (ngx_conf_full_name(cycle, &cycle->conf_file, 0) != NGX_OK) { |
| return NGX_ERROR; |
| } |
| |
| for (p = cycle->conf_file.data + cycle->conf_file.len - 1; |
| p > cycle->conf_file.data; |
| p--) |
| { |
| if (ngx_path_separator(*p)) { |
| cycle->conf_prefix.len = p - cycle->conf_file.data + 1; |
| cycle->conf_prefix.data = cycle->conf_file.data; |
| break; |
| } |
| } |
| |
| if (ngx_error_log) { |
| cycle->error_log.len = ngx_strlen(ngx_error_log); |
| cycle->error_log.data = ngx_error_log; |
| |
| } else { |
| ngx_str_set(&cycle->error_log, NGX_ERROR_LOG_PATH); |
| } |
| |
| if (ngx_conf_params) { |
| cycle->conf_param.len = ngx_strlen(ngx_conf_params); |
| cycle->conf_param.data = ngx_conf_params; |
| } |
| |
| if (ngx_test_config) { |
| cycle->log->log_level = NGX_LOG_INFO; |
| } |
| |
| return NGX_OK; |
| } |
| |
| |
| static void * |
| ngx_core_module_create_conf(ngx_cycle_t *cycle) |
| { |
| ngx_core_conf_t *ccf; |
| |
| ccf = ngx_pcalloc(cycle->pool, sizeof(ngx_core_conf_t)); |
| if (ccf == NULL) { |
| return NULL; |
| } |
| |
| /* |
| * set by ngx_pcalloc() |
| * |
| * ccf->pid = NULL; |
| * ccf->oldpid = NULL; |
| * ccf->priority = 0; |
| * ccf->cpu_affinity_auto = 0; |
| * ccf->cpu_affinity_n = 0; |
| * ccf->cpu_affinity = NULL; |
| */ |
| |
| ccf->daemon = NGX_CONF_UNSET; |
| ccf->master = NGX_CONF_UNSET; |
| ccf->timer_resolution = NGX_CONF_UNSET_MSEC; |
| ccf->shutdown_timeout = NGX_CONF_UNSET_MSEC; |
| |
| ccf->worker_processes = NGX_CONF_UNSET; |
| ccf->debug_points = NGX_CONF_UNSET; |
| |
| ccf->rlimit_nofile = NGX_CONF_UNSET; |
| ccf->rlimit_core = NGX_CONF_UNSET; |
| |
| ccf->user = (ngx_uid_t) NGX_CONF_UNSET_UINT; |
| ccf->group = (ngx_gid_t) NGX_CONF_UNSET_UINT; |
| |
| if (ngx_array_init(&ccf->env, cycle->pool, 1, sizeof(ngx_str_t)) |
| != NGX_OK) |
| { |
| return NULL; |
| } |
| |
| return ccf; |
| } |
| |
| |
| static char * |
| ngx_core_module_init_conf(ngx_cycle_t *cycle, void *conf) |
| { |
| ngx_core_conf_t *ccf = conf; |
| |
| ngx_conf_init_value(ccf->daemon, 1); |
| ngx_conf_init_value(ccf->master, 1); |
| ngx_conf_init_msec_value(ccf->timer_resolution, 0); |
| ngx_conf_init_msec_value(ccf->shutdown_timeout, 0); |
| |
| ngx_conf_init_value(ccf->worker_processes, 1); |
| ngx_conf_init_value(ccf->debug_points, 0); |
| |
| #if (NGX_HAVE_CPU_AFFINITY) |
| |
| if (!ccf->cpu_affinity_auto |
| && ccf->cpu_affinity_n |
| && ccf->cpu_affinity_n != 1 |
| && ccf->cpu_affinity_n != (ngx_uint_t) ccf->worker_processes) |
| { |
| ngx_log_error(NGX_LOG_WARN, cycle->log, 0, |
| "the number of \"worker_processes\" is not equal to " |
| "the number of \"worker_cpu_affinity\" masks, " |
| "using last mask for remaining worker processes"); |
| } |
| |
| #endif |
| |
| |
| if (ccf->pid.len == 0) { |
| ngx_str_set(&ccf->pid, NGX_PID_PATH); |
| } |
| |
| if (ngx_conf_full_name(cycle, &ccf->pid, 0) != NGX_OK) { |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->oldpid.len = ccf->pid.len + sizeof(NGX_OLDPID_EXT); |
| |
| ccf->oldpid.data = ngx_pnalloc(cycle->pool, ccf->oldpid.len); |
| if (ccf->oldpid.data == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| ngx_memcpy(ngx_cpymem(ccf->oldpid.data, ccf->pid.data, ccf->pid.len), |
| NGX_OLDPID_EXT, sizeof(NGX_OLDPID_EXT)); |
| |
| |
| #if !(NGX_WIN32) |
| |
| if (ccf->user == (uid_t) NGX_CONF_UNSET_UINT && geteuid() == 0) { |
| struct group *grp; |
| struct passwd *pwd; |
| |
| ngx_set_errno(0); |
| pwd = getpwnam(NGX_USER); |
| if (pwd == NULL) { |
| ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno, |
| "getpwnam(\"" NGX_USER "\") failed"); |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->username = NGX_USER; |
| ccf->user = pwd->pw_uid; |
| |
| ngx_set_errno(0); |
| grp = getgrnam(NGX_GROUP); |
| if (grp == NULL) { |
| ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno, |
| "getgrnam(\"" NGX_GROUP "\") failed"); |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->group = grp->gr_gid; |
| } |
| |
| |
| if (ccf->lock_file.len == 0) { |
| ngx_str_set(&ccf->lock_file, NGX_LOCK_PATH); |
| } |
| |
| if (ngx_conf_full_name(cycle, &ccf->lock_file, 0) != NGX_OK) { |
| return NGX_CONF_ERROR; |
| } |
| |
| { |
| ngx_str_t lock_file; |
| |
| lock_file = cycle->old_cycle->lock_file; |
| |
| if (lock_file.len) { |
| lock_file.len--; |
| |
| if (ccf->lock_file.len != lock_file.len |
| || ngx_strncmp(ccf->lock_file.data, lock_file.data, lock_file.len) |
| != 0) |
| { |
| ngx_log_error(NGX_LOG_EMERG, cycle->log, 0, |
| "\"lock_file\" could not be changed, ignored"); |
| } |
| |
| cycle->lock_file.len = lock_file.len + 1; |
| lock_file.len += sizeof(".accept"); |
| |
| cycle->lock_file.data = ngx_pstrdup(cycle->pool, &lock_file); |
| if (cycle->lock_file.data == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| } else { |
| cycle->lock_file.len = ccf->lock_file.len + 1; |
| cycle->lock_file.data = ngx_pnalloc(cycle->pool, |
| ccf->lock_file.len + sizeof(".accept")); |
| if (cycle->lock_file.data == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| ngx_memcpy(ngx_cpymem(cycle->lock_file.data, ccf->lock_file.data, |
| ccf->lock_file.len), |
| ".accept", sizeof(".accept")); |
| } |
| } |
| |
| #endif |
| |
| return NGX_CONF_OK; |
| } |
| |
| |
| static char * |
| ngx_set_user(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| #if (NGX_WIN32) |
| |
| ngx_conf_log_error(NGX_LOG_WARN, cf, 0, |
| "\"user\" is not supported, ignored"); |
| |
| return NGX_CONF_OK; |
| |
| #else |
| |
| ngx_core_conf_t *ccf = conf; |
| |
| char *group; |
| struct passwd *pwd; |
| struct group *grp; |
| ngx_str_t *value; |
| |
| if (ccf->user != (uid_t) NGX_CONF_UNSET_UINT) { |
| return "is duplicate"; |
| } |
| |
| if (geteuid() != 0) { |
| ngx_conf_log_error(NGX_LOG_WARN, cf, 0, |
| "the \"user\" directive makes sense only " |
| "if the master process runs " |
| "with super-user privileges, ignored"); |
| return NGX_CONF_OK; |
| } |
| |
| value = cf->args->elts; |
| |
| ccf->username = (char *) value[1].data; |
| |
| ngx_set_errno(0); |
| pwd = getpwnam((const char *) value[1].data); |
| if (pwd == NULL) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, ngx_errno, |
| "getpwnam(\"%s\") failed", value[1].data); |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->user = pwd->pw_uid; |
| |
| group = (char *) ((cf->args->nelts == 2) ? value[1].data : value[2].data); |
| |
| ngx_set_errno(0); |
| grp = getgrnam(group); |
| if (grp == NULL) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, ngx_errno, |
| "getgrnam(\"%s\") failed", group); |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->group = grp->gr_gid; |
| |
| return NGX_CONF_OK; |
| |
| #endif |
| } |
| |
| |
| static char * |
| ngx_set_env(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| ngx_core_conf_t *ccf = conf; |
| |
| ngx_str_t *value, *var; |
| ngx_uint_t i; |
| |
| var = ngx_array_push(&ccf->env); |
| if (var == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| value = cf->args->elts; |
| *var = value[1]; |
| |
| for (i = 0; i < value[1].len; i++) { |
| |
| if (value[1].data[i] == '=') { |
| |
| var->len = i; |
| |
| return NGX_CONF_OK; |
| } |
| } |
| |
| return NGX_CONF_OK; |
| } |
| |
| |
| static char * |
| ngx_set_priority(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| ngx_core_conf_t *ccf = conf; |
| |
| ngx_str_t *value; |
| ngx_uint_t n, minus; |
| |
| if (ccf->priority != 0) { |
| return "is duplicate"; |
| } |
| |
| value = cf->args->elts; |
| |
| if (value[1].data[0] == '-') { |
| n = 1; |
| minus = 1; |
| |
| } else if (value[1].data[0] == '+') { |
| n = 1; |
| minus = 0; |
| |
| } else { |
| n = 0; |
| minus = 0; |
| } |
| |
| ccf->priority = ngx_atoi(&value[1].data[n], value[1].len - n); |
| if (ccf->priority == NGX_ERROR) { |
| return "invalid number"; |
| } |
| |
| if (minus) { |
| ccf->priority = -ccf->priority; |
| } |
| |
| return NGX_CONF_OK; |
| } |
| |
| |
| static char * |
| ngx_set_cpu_affinity(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| #if (NGX_HAVE_CPU_AFFINITY) |
| ngx_core_conf_t *ccf = conf; |
| |
| u_char ch, *p; |
| ngx_str_t *value; |
| ngx_uint_t i, n; |
| ngx_cpuset_t *mask; |
| |
| if (ccf->cpu_affinity) { |
| return "is duplicate"; |
| } |
| |
| mask = ngx_palloc(cf->pool, (cf->args->nelts - 1) * sizeof(ngx_cpuset_t)); |
| if (mask == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->cpu_affinity_n = cf->args->nelts - 1; |
| ccf->cpu_affinity = mask; |
| |
| value = cf->args->elts; |
| |
| if (ngx_strcmp(value[1].data, "auto") == 0) { |
| |
| if (cf->args->nelts > 3) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| "invalid number of arguments in " |
| "\"worker_cpu_affinity\" directive"); |
| return NGX_CONF_ERROR; |
| } |
| |
| ccf->cpu_affinity_auto = 1; |
| |
| CPU_ZERO(&mask[0]); |
| for (i = 0; i < (ngx_uint_t) ngx_min(ngx_ncpu, CPU_SETSIZE); i++) { |
| CPU_SET(i, &mask[0]); |
| } |
| |
| n = 2; |
| |
| } else { |
| n = 1; |
| } |
| |
| for ( /* void */ ; n < cf->args->nelts; n++) { |
| |
| if (value[n].len > CPU_SETSIZE) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| "\"worker_cpu_affinity\" supports up to %d CPUs only", |
| CPU_SETSIZE); |
| return NGX_CONF_ERROR; |
| } |
| |
| i = 0; |
| CPU_ZERO(&mask[n - 1]); |
| |
| for (p = value[n].data + value[n].len - 1; |
| p >= value[n].data; |
| p--) |
| { |
| ch = *p; |
| |
| if (ch == ' ') { |
| continue; |
| } |
| |
| i++; |
| |
| if (ch == '0') { |
| continue; |
| } |
| |
| if (ch == '1') { |
| CPU_SET(i - 1, &mask[n - 1]); |
| continue; |
| } |
| |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| "invalid character \"%c\" in \"worker_cpu_affinity\"", |
| ch); |
| return NGX_CONF_ERROR; |
| } |
| } |
| |
| #else |
| |
| ngx_conf_log_error(NGX_LOG_WARN, cf, 0, |
| "\"worker_cpu_affinity\" is not supported " |
| "on this platform, ignored"); |
| #endif |
| |
| return NGX_CONF_OK; |
| } |
| |
| |
| ngx_cpuset_t * |
| ngx_get_cpu_affinity(ngx_uint_t n) |
| { |
| #if (NGX_HAVE_CPU_AFFINITY) |
| ngx_uint_t i, j; |
| ngx_cpuset_t *mask; |
| ngx_core_conf_t *ccf; |
| |
| static ngx_cpuset_t result; |
| |
| ccf = (ngx_core_conf_t *) ngx_get_conf(ngx_cycle->conf_ctx, |
| ngx_core_module); |
| |
| if (ccf->cpu_affinity == NULL) { |
| return NULL; |
| } |
| |
| if (ccf->cpu_affinity_auto) { |
| mask = &ccf->cpu_affinity[ccf->cpu_affinity_n - 1]; |
| |
| for (i = 0, j = n; /* void */ ; i++) { |
| |
| if (CPU_ISSET(i % CPU_SETSIZE, mask) && j-- == 0) { |
| break; |
| } |
| |
| if (i == CPU_SETSIZE && j == n) { |
| /* empty mask */ |
| return NULL; |
| } |
| |
| /* void */ |
| } |
| |
| CPU_ZERO(&result); |
| CPU_SET(i % CPU_SETSIZE, &result); |
| |
| return &result; |
| } |
| |
| if (ccf->cpu_affinity_n > n) { |
| return &ccf->cpu_affinity[n]; |
| } |
| |
| return &ccf->cpu_affinity[ccf->cpu_affinity_n - 1]; |
| |
| #else |
| |
| return NULL; |
| |
| #endif |
| } |
| |
| |
| static char * |
| ngx_set_worker_processes(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| ngx_str_t *value; |
| ngx_core_conf_t *ccf; |
| |
| ccf = (ngx_core_conf_t *) conf; |
| |
| if (ccf->worker_processes != NGX_CONF_UNSET) { |
| return "is duplicate"; |
| } |
| |
| value = cf->args->elts; |
| |
| if (ngx_strcmp(value[1].data, "auto") == 0) { |
| ccf->worker_processes = ngx_ncpu; |
| return NGX_CONF_OK; |
| } |
| |
| ccf->worker_processes = ngx_atoi(value[1].data, value[1].len); |
| |
| if (ccf->worker_processes == NGX_ERROR) { |
| return "invalid value"; |
| } |
| |
| return NGX_CONF_OK; |
| } |
| |
| |
| static char * |
| ngx_load_module(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| #if (NGX_HAVE_DLOPEN) |
| void *handle; |
| char **names, **order; |
| ngx_str_t *value, file; |
| ngx_uint_t i; |
| ngx_module_t *module, **modules; |
| ngx_pool_cleanup_t *cln; |
| |
| if (cf->cycle->modules_used) { |
| return "is specified too late"; |
| } |
| |
| value = cf->args->elts; |
| |
| file = value[1]; |
| |
| if (ngx_conf_full_name(cf->cycle, &file, 0) != NGX_OK) { |
| return NGX_CONF_ERROR; |
| } |
| |
| cln = ngx_pool_cleanup_add(cf->cycle->pool, 0); |
| if (cln == NULL) { |
| return NGX_CONF_ERROR; |
| } |
| |
| handle = ngx_dlopen(file.data); |
| if (handle == NULL) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| ngx_dlopen_n " \"%s\" failed (%s)", |
| file.data, ngx_dlerror()); |
| return NGX_CONF_ERROR; |
| } |
| |
| cln->handler = ngx_unload_module; |
| cln->data = handle; |
| |
| modules = ngx_dlsym(handle, "ngx_modules"); |
| if (modules == NULL) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| ngx_dlsym_n " \"%V\", \"%s\" failed (%s)", |
| &value[1], "ngx_modules", ngx_dlerror()); |
| return NGX_CONF_ERROR; |
| } |
| |
| names = ngx_dlsym(handle, "ngx_module_names"); |
| if (names == NULL) { |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| ngx_dlsym_n " \"%V\", \"%s\" failed (%s)", |
| &value[1], "ngx_module_names", ngx_dlerror()); |
| return NGX_CONF_ERROR; |
| } |
| |
| order = ngx_dlsym(handle, "ngx_module_order"); |
| |
| for (i = 0; modules[i]; i++) { |
| module = modules[i]; |
| module->name = names[i]; |
| |
| if (ngx_add_module(cf, &file, module, order) != NGX_OK) { |
| return NGX_CONF_ERROR; |
| } |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_CORE, cf->log, 0, "module: %s i:%ui", |
| module->name, module->index); |
| } |
| |
| return NGX_CONF_OK; |
| |
| #else |
| |
| ngx_conf_log_error(NGX_LOG_EMERG, cf, 0, |
| "\"load_module\" is not supported " |
| "on this platform"); |
| return NGX_CONF_ERROR; |
| |
| #endif |
| } |
| |
| |
| #if (NGX_HAVE_DLOPEN) |
| |
| static void |
| ngx_unload_module(void *data) |
| { |
| void *handle = data; |
| |
| if (ngx_dlclose(handle) != 0) { |
| ngx_log_error(NGX_LOG_ALERT, ngx_cycle->log, 0, |
| ngx_dlclose_n " failed (%s)", ngx_dlerror()); |
| } |
| } |
| |
| #endif |