src/os/unix/ngx_process_cycle.c - nginx-1.7.10

Global variables defined

Functions defined

Source code


  1. /*
  2. * Copyright (C) Igor Sysoev
  3. * Copyright (C) Nginx, Inc.
  4. */


  5. #include <ngx_config.h>
  6. #include <ngx_core.h>
  7. #include <ngx_event.h>
  8. #include <ngx_channel.h>


  9. static void ngx_start_worker_processes(ngx_cycle_t *cycle, ngx_int_t n,
  10.     ngx_int_t type);
  11. static void ngx_start_cache_manager_processes(ngx_cycle_t *cycle,
  12.     ngx_uint_t respawn);
  13. static void ngx_pass_open_channel(ngx_cycle_t *cycle, ngx_channel_t *ch);
  14. static void ngx_signal_worker_processes(ngx_cycle_t *cycle, int signo);
  15. static ngx_uint_t ngx_reap_children(ngx_cycle_t *cycle);
  16. static void ngx_master_process_exit(ngx_cycle_t *cycle);
  17. static void ngx_worker_process_cycle(ngx_cycle_t *cycle, void *data);
  18. static void ngx_worker_process_init(ngx_cycle_t *cycle, ngx_int_t worker);
  19. static void ngx_worker_process_exit(ngx_cycle_t *cycle);
  20. static void ngx_channel_handler(ngx_event_t *ev);
  21. #if (NGX_THREADS)
  22. static void ngx_wakeup_worker_threads(ngx_cycle_t *cycle);
  23. static ngx_thread_value_t ngx_worker_thread_cycle(void *data);
  24. #endif
  25. static void ngx_cache_manager_process_cycle(ngx_cycle_t *cycle, void *data);
  26. static void ngx_cache_manager_process_handler(ngx_event_t *ev);
  27. static void ngx_cache_loader_process_handler(ngx_event_t *ev);


  28. ngx_uint_t    ngx_process;
  29. ngx_pid_t     ngx_pid;
  30. ngx_uint_t    ngx_threaded;

  31. sig_atomic_t  ngx_reap;
  32. sig_atomic_t  ngx_sigio;
  33. sig_atomic_t  ngx_sigalrm;
  34. sig_atomic_t  ngx_terminate;
  35. sig_atomic_t  ngx_quit;
  36. sig_atomic_t  ngx_debug_quit;
  37. ngx_uint_t    ngx_exiting;
  38. sig_atomic_t  ngx_reconfigure;
  39. sig_atomic_t  ngx_reopen;

  40. sig_atomic_t  ngx_change_binary;
  41. ngx_pid_t     ngx_new_binary;
  42. ngx_uint_t    ngx_inherited;
  43. ngx_uint_t    ngx_daemonized;

  44. sig_atomic_t  ngx_noaccept;
  45. ngx_uint_t    ngx_noaccepting;
  46. ngx_uint_t    ngx_restart;


  47. #if (NGX_THREADS)
  48. volatile ngx_thread_t  ngx_threads[NGX_MAX_THREADS];
  49. ngx_int_t              ngx_threads_n;
  50. #endif


  51. static u_char  master_process[] = "master process";


  52. static ngx_cache_manager_ctx_t  ngx_cache_manager_ctx = {
  53.     ngx_cache_manager_process_handler, "cache manager process", 0
  54. };

  55. static ngx_cache_manager_ctx_t  ngx_cache_loader_ctx = {
  56.     ngx_cache_loader_process_handler, "cache loader process", 60000
  57. };


  58. static ngx_cycle_t      ngx_exit_cycle;
  59. static ngx_log_t        ngx_exit_log;
  60. static ngx_open_file_t  ngx_exit_log_file;


  61. void
  62. ngx_master_process_cycle(ngx_cycle_t *cycle)
  63. {
  64.     char              *title;
  65.     u_char            *p;
  66.     size_t             size;
  67.     ngx_int_t          i;
  68.     ngx_uint_t         n, sigio;
  69.     sigset_t           set;
  70.     struct itimerval   itv;
  71.     ngx_uint_t         live;
  72.     ngx_msec_t         delay;
  73.     ngx_listening_t   *ls;
  74.     ngx_core_conf_t   *ccf;

  75.     sigemptyset(&set);
  76.     sigaddset(&set, SIGCHLD);
  77.     sigaddset(&set, SIGALRM);
  78.     sigaddset(&set, SIGIO);
  79.     sigaddset(&set, SIGINT);
  80.     sigaddset(&set, ngx_signal_value(NGX_RECONFIGURE_SIGNAL));
  81.     sigaddset(&set, ngx_signal_value(NGX_REOPEN_SIGNAL));
  82.     sigaddset(&set, ngx_signal_value(NGX_NOACCEPT_SIGNAL));
  83.     sigaddset(&set, ngx_signal_value(NGX_TERMINATE_SIGNAL));
  84.     sigaddset(&set, ngx_signal_value(NGX_SHUTDOWN_SIGNAL));
  85.     sigaddset(&set, ngx_signal_value(NGX_CHANGEBIN_SIGNAL));

  86.     if (sigprocmask(SIG_BLOCK, &set, NULL) == -1) {
  87.         ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  88.                       "sigprocmask() failed");
  89.     }

  90.     sigemptyset(&set);


  91.     size = sizeof(master_process);

  92.     for (i = 0; i < ngx_argc; i++) {
  93.         size += ngx_strlen(ngx_argv[i]) + 1;
  94.     }

  95.     title = ngx_pnalloc(cycle->pool, size);
  96.     if (title == NULL) {
  97.         /* fatal */
  98.         exit(2);
  99.     }

  100.     p = ngx_cpymem(title, master_process, sizeof(master_process) - 1);
  101.     for (i = 0; i < ngx_argc; i++) {
  102.         *p++ = ' ';
  103.         p = ngx_cpystrn(p, (u_char *) ngx_argv[i], size);
  104.     }

  105.     ngx_setproctitle(title);


  106.     ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);

  107.     ngx_start_worker_processes(cycle, ccf->worker_processes,
  108.                                NGX_PROCESS_RESPAWN);
  109.     ngx_start_cache_manager_processes(cycle, 0);

  110.     ngx_new_binary = 0;
  111.     delay = 0;
  112.     sigio = 0;
  113.     live = 1;

  114.     for ( ;; ) {
  115.         if (delay) {
  116.             if (ngx_sigalrm) {
  117.                 sigio = 0;
  118.                 delay *= 2;
  119.                 ngx_sigalrm = 0;
  120.             }

  121.             ngx_log_debug1(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  122.                            "termination cycle: %d", delay);

  123.             itv.it_interval.tv_sec = 0;
  124.             itv.it_interval.tv_usec = 0;
  125.             itv.it_value.tv_sec = delay / 1000;
  126.             itv.it_value.tv_usec = (delay % 1000 ) * 1000;

  127.             if (setitimer(ITIMER_REAL, &itv, NULL) == -1) {
  128.                 ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  129.                               "setitimer() failed");
  130.             }
  131.         }

  132.         ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, "sigsuspend");

  133.         sigsuspend(&set);

  134.         ngx_time_update();

  135.         ngx_log_debug1(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  136.                        "wake up, sigio %i", sigio);

  137.         if (ngx_reap) {
  138.             ngx_reap = 0;
  139.             ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, "reap children");

  140.             live = ngx_reap_children(cycle);
  141.         }

  142.         if (!live && (ngx_terminate || ngx_quit)) {
  143.             ngx_master_process_exit(cycle);
  144.         }

  145.         if (ngx_terminate) {
  146.             if (delay == 0) {
  147.                 delay = 50;
  148.             }

  149.             if (sigio) {
  150.                 sigio--;
  151.                 continue;
  152.             }

  153.             sigio = ccf->worker_processes + 2 /* cache processes */;

  154.             if (delay > 1000) {
  155.                 ngx_signal_worker_processes(cycle, SIGKILL);
  156.             } else {
  157.                 ngx_signal_worker_processes(cycle,
  158.                                        ngx_signal_value(NGX_TERMINATE_SIGNAL));
  159.             }

  160.             continue;
  161.         }

  162.         if (ngx_quit) {
  163.             ngx_signal_worker_processes(cycle,
  164.                                         ngx_signal_value(NGX_SHUTDOWN_SIGNAL));

  165.             ls = cycle->listening.elts;
  166.             for (n = 0; n < cycle->listening.nelts; n++) {
  167.                 if (ngx_close_socket(ls[n].fd) == -1) {
  168.                     ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_socket_errno,
  169.                                   ngx_close_socket_n " %V failed",
  170.                                   &ls[n].addr_text);
  171.                 }
  172.             }
  173.             cycle->listening.nelts = 0;

  174.             continue;
  175.         }

  176.         if (ngx_reconfigure) {
  177.             ngx_reconfigure = 0;

  178.             if (ngx_new_binary) {
  179.                 ngx_start_worker_processes(cycle, ccf->worker_processes,
  180.                                            NGX_PROCESS_RESPAWN);
  181.                 ngx_start_cache_manager_processes(cycle, 0);
  182.                 ngx_noaccepting = 0;

  183.                 continue;
  184.             }

  185.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reconfiguring");

  186.             cycle = ngx_init_cycle(cycle);
  187.             if (cycle == NULL) {
  188.                 cycle = (ngx_cycle_t *) ngx_cycle;
  189.                 continue;
  190.             }

  191.             ngx_cycle = cycle;
  192.             ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx,
  193.                                                    ngx_core_module);
  194.             ngx_start_worker_processes(cycle, ccf->worker_processes,
  195.                                        NGX_PROCESS_JUST_RESPAWN);
  196.             ngx_start_cache_manager_processes(cycle, 1);

  197.             /* allow new processes to start */
  198.             ngx_msleep(100);

  199.             live = 1;
  200.             ngx_signal_worker_processes(cycle,
  201.                                         ngx_signal_value(NGX_SHUTDOWN_SIGNAL));
  202.         }

  203.         if (ngx_restart) {
  204.             ngx_restart = 0;
  205.             ngx_start_worker_processes(cycle, ccf->worker_processes,
  206.                                        NGX_PROCESS_RESPAWN);
  207.             ngx_start_cache_manager_processes(cycle, 0);
  208.             live = 1;
  209.         }

  210.         if (ngx_reopen) {
  211.             ngx_reopen = 0;
  212.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reopening logs");
  213.             ngx_reopen_files(cycle, ccf->user);
  214.             ngx_signal_worker_processes(cycle,
  215.                                         ngx_signal_value(NGX_REOPEN_SIGNAL));
  216.         }

  217.         if (ngx_change_binary) {
  218.             ngx_change_binary = 0;
  219.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "changing binary");
  220.             ngx_new_binary = ngx_exec_new_binary(cycle, ngx_argv);
  221.         }

  222.         if (ngx_noaccept) {
  223.             ngx_noaccept = 0;
  224.             ngx_noaccepting = 1;
  225.             ngx_signal_worker_processes(cycle,
  226.                                         ngx_signal_value(NGX_SHUTDOWN_SIGNAL));
  227.         }
  228.     }
  229. }


  230. void
  231. ngx_single_process_cycle(ngx_cycle_t *cycle)
  232. {
  233.     ngx_uint_t  i;

  234.     if (ngx_set_environment(cycle, NULL) == NULL) {
  235.         /* fatal */
  236.         exit(2);
  237.     }

  238.     for (i = 0; ngx_modules[i]; i++) {
  239.         if (ngx_modules[i]->init_process) {
  240.             if (ngx_modules[i]->init_process(cycle) == NGX_ERROR) {
  241.                 /* fatal */
  242.                 exit(2);
  243.             }
  244.         }
  245.     }

  246.     for ( ;; ) {
  247.         ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, "worker cycle");

  248.         ngx_process_events_and_timers(cycle);

  249.         if (ngx_terminate || ngx_quit) {

  250.             for (i = 0; ngx_modules[i]; i++) {
  251.                 if (ngx_modules[i]->exit_process) {
  252.                     ngx_modules[i]->exit_process(cycle);
  253.                 }
  254.             }

  255.             ngx_master_process_exit(cycle);
  256.         }

  257.         if (ngx_reconfigure) {
  258.             ngx_reconfigure = 0;
  259.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reconfiguring");

  260.             cycle = ngx_init_cycle(cycle);
  261.             if (cycle == NULL) {
  262.                 cycle = (ngx_cycle_t *) ngx_cycle;
  263.                 continue;
  264.             }

  265.             ngx_cycle = cycle;
  266.         }

  267.         if (ngx_reopen) {
  268.             ngx_reopen = 0;
  269.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reopening logs");
  270.             ngx_reopen_files(cycle, (ngx_uid_t) -1);
  271.         }
  272.     }
  273. }


  274. static void
  275. ngx_start_worker_processes(ngx_cycle_t *cycle, ngx_int_t n, ngx_int_t type)
  276. {
  277.     ngx_int_t      i;
  278.     ngx_channel_t  ch;

  279.     ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "start worker processes");

  280.     ngx_memzero(&ch, sizeof(ngx_channel_t));

  281.     ch.command = NGX_CMD_OPEN_CHANNEL;

  282.     for (i = 0; i < n; i++) {

  283.         ngx_spawn_process(cycle, ngx_worker_process_cycle,
  284.                           (void *) (intptr_t) i, "worker process", type);

  285.         ch.pid = ngx_processes[ngx_process_slot].pid;
  286.         ch.slot = ngx_process_slot;
  287.         ch.fd = ngx_processes[ngx_process_slot].channel[0];

  288.         ngx_pass_open_channel(cycle, &ch);
  289.     }
  290. }


  291. static void
  292. ngx_start_cache_manager_processes(ngx_cycle_t *cycle, ngx_uint_t respawn)
  293. {
  294.     ngx_uint_t       i, manager, loader;
  295.     ngx_path_t     **path;
  296.     ngx_channel_t    ch;

  297.     manager = 0;
  298.     loader = 0;

  299.     path = ngx_cycle->paths.elts;
  300.     for (i = 0; i < ngx_cycle->paths.nelts; i++) {

  301.         if (path[i]->manager) {
  302.             manager = 1;
  303.         }

  304.         if (path[i]->loader) {
  305.             loader = 1;
  306.         }
  307.     }

  308.     if (manager == 0) {
  309.         return;
  310.     }

  311.     ngx_spawn_process(cycle, ngx_cache_manager_process_cycle,
  312.                       &ngx_cache_manager_ctx, "cache manager process",
  313.                       respawn ? NGX_PROCESS_JUST_RESPAWN : NGX_PROCESS_RESPAWN);

  314.     ngx_memzero(&ch, sizeof(ngx_channel_t));

  315.     ch.command = NGX_CMD_OPEN_CHANNEL;
  316.     ch.pid = ngx_processes[ngx_process_slot].pid;
  317.     ch.slot = ngx_process_slot;
  318.     ch.fd = ngx_processes[ngx_process_slot].channel[0];

  319.     ngx_pass_open_channel(cycle, &ch);

  320.     if (loader == 0) {
  321.         return;
  322.     }

  323.     ngx_spawn_process(cycle, ngx_cache_manager_process_cycle,
  324.                       &ngx_cache_loader_ctx, "cache loader process",
  325.                       respawn ? NGX_PROCESS_JUST_SPAWN : NGX_PROCESS_NORESPAWN);

  326.     ch.command = NGX_CMD_OPEN_CHANNEL;
  327.     ch.pid = ngx_processes[ngx_process_slot].pid;
  328.     ch.slot = ngx_process_slot;
  329.     ch.fd = ngx_processes[ngx_process_slot].channel[0];

  330.     ngx_pass_open_channel(cycle, &ch);
  331. }


  332. static void
  333. ngx_pass_open_channel(ngx_cycle_t *cycle, ngx_channel_t *ch)
  334. {
  335.     ngx_int_t  i;

  336.     for (i = 0; i < ngx_last_process; i++) {

  337.         if (i == ngx_process_slot
  338.             || ngx_processes[i].pid == -1
  339.             || ngx_processes[i].channel[0] == -1)
  340.         {
  341.             continue;
  342.         }

  343.         ngx_log_debug6(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  344.                       "pass channel s:%d pid:%P fd:%d to s:%i pid:%P fd:%d",
  345.                       ch->slot, ch->pid, ch->fd,
  346.                       i, ngx_processes[i].pid,
  347.                       ngx_processes[i].channel[0]);

  348.         /* TODO: NGX_AGAIN */

  349.         ngx_write_channel(ngx_processes[i].channel[0],
  350.                           ch, sizeof(ngx_channel_t), cycle->log);
  351.     }
  352. }


  353. static void
  354. ngx_signal_worker_processes(ngx_cycle_t *cycle, int signo)
  355. {
  356.     ngx_int_t      i;
  357.     ngx_err_t      err;
  358.     ngx_channel_t  ch;

  359.     ngx_memzero(&ch, sizeof(ngx_channel_t));

  360. #if (NGX_BROKEN_SCM_RIGHTS)

  361.     ch.command = 0;

  362. #else

  363.     switch (signo) {

  364.     case ngx_signal_value(NGX_SHUTDOWN_SIGNAL):
  365.         ch.command = NGX_CMD_QUIT;
  366.         break;

  367.     case ngx_signal_value(NGX_TERMINATE_SIGNAL):
  368.         ch.command = NGX_CMD_TERMINATE;
  369.         break;

  370.     case ngx_signal_value(NGX_REOPEN_SIGNAL):
  371.         ch.command = NGX_CMD_REOPEN;
  372.         break;

  373.     default:
  374.         ch.command = 0;
  375.     }

  376. #endif

  377.     ch.fd = -1;


  378.     for (i = 0; i < ngx_last_process; i++) {

  379.         ngx_log_debug7(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  380.                        "child: %d %P e:%d t:%d d:%d r:%d j:%d",
  381.                        i,
  382.                        ngx_processes[i].pid,
  383.                        ngx_processes[i].exiting,
  384.                        ngx_processes[i].exited,
  385.                        ngx_processes[i].detached,
  386.                        ngx_processes[i].respawn,
  387.                        ngx_processes[i].just_spawn);

  388.         if (ngx_processes[i].detached || ngx_processes[i].pid == -1) {
  389.             continue;
  390.         }

  391.         if (ngx_processes[i].just_spawn) {
  392.             ngx_processes[i].just_spawn = 0;
  393.             continue;
  394.         }

  395.         if (ngx_processes[i].exiting
  396.             && signo == ngx_signal_value(NGX_SHUTDOWN_SIGNAL))
  397.         {
  398.             continue;
  399.         }

  400.         if (ch.command) {
  401.             if (ngx_write_channel(ngx_processes[i].channel[0],
  402.                                   &ch, sizeof(ngx_channel_t), cycle->log)
  403.                 == NGX_OK)
  404.             {
  405.                 if (signo != ngx_signal_value(NGX_REOPEN_SIGNAL)) {
  406.                     ngx_processes[i].exiting = 1;
  407.                 }

  408.                 continue;
  409.             }
  410.         }

  411.         ngx_log_debug2(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  412.                        "kill (%P, %d)", ngx_processes[i].pid, signo);

  413.         if (kill(ngx_processes[i].pid, signo) == -1) {
  414.             err = ngx_errno;
  415.             ngx_log_error(NGX_LOG_ALERT, cycle->log, err,
  416.                           "kill(%P, %d) failed", ngx_processes[i].pid, signo);

  417.             if (err == NGX_ESRCH) {
  418.                 ngx_processes[i].exited = 1;
  419.                 ngx_processes[i].exiting = 0;
  420.                 ngx_reap = 1;
  421.             }

  422.             continue;
  423.         }

  424.         if (signo != ngx_signal_value(NGX_REOPEN_SIGNAL)) {
  425.             ngx_processes[i].exiting = 1;
  426.         }
  427.     }
  428. }


  429. static ngx_uint_t
  430. ngx_reap_children(ngx_cycle_t *cycle)
  431. {
  432.     ngx_int_t         i, n;
  433.     ngx_uint_t        live;
  434.     ngx_channel_t     ch;
  435.     ngx_core_conf_t  *ccf;

  436.     ngx_memzero(&ch, sizeof(ngx_channel_t));

  437.     ch.command = NGX_CMD_CLOSE_CHANNEL;
  438.     ch.fd = -1;

  439.     live = 0;
  440.     for (i = 0; i < ngx_last_process; i++) {

  441.         ngx_log_debug7(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  442.                        "child: %d %P e:%d t:%d d:%d r:%d j:%d",
  443.                        i,
  444.                        ngx_processes[i].pid,
  445.                        ngx_processes[i].exiting,
  446.                        ngx_processes[i].exited,
  447.                        ngx_processes[i].detached,
  448.                        ngx_processes[i].respawn,
  449.                        ngx_processes[i].just_spawn);

  450.         if (ngx_processes[i].pid == -1) {
  451.             continue;
  452.         }

  453.         if (ngx_processes[i].exited) {

  454.             if (!ngx_processes[i].detached) {
  455.                 ngx_close_channel(ngx_processes[i].channel, cycle->log);

  456.                 ngx_processes[i].channel[0] = -1;
  457.                 ngx_processes[i].channel[1] = -1;

  458.                 ch.pid = ngx_processes[i].pid;
  459.                 ch.slot = i;

  460.                 for (n = 0; n < ngx_last_process; n++) {
  461.                     if (ngx_processes[n].exited
  462.                         || ngx_processes[n].pid == -1
  463.                         || ngx_processes[n].channel[0] == -1)
  464.                     {
  465.                         continue;
  466.                     }

  467.                     ngx_log_debug3(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  468.                                    "pass close channel s:%i pid:%P to:%P",
  469.                                    ch.slot, ch.pid, ngx_processes[n].pid);

  470.                     /* TODO: NGX_AGAIN */

  471.                     ngx_write_channel(ngx_processes[n].channel[0],
  472.                                       &ch, sizeof(ngx_channel_t), cycle->log);
  473.                 }
  474.             }

  475.             if (ngx_processes[i].respawn
  476.                 && !ngx_processes[i].exiting
  477.                 && !ngx_terminate
  478.                 && !ngx_quit)
  479.             {
  480.                 if (ngx_spawn_process(cycle, ngx_processes[i].proc,
  481.                                       ngx_processes[i].data,
  482.                                       ngx_processes[i].name, i)
  483.                     == NGX_INVALID_PID)
  484.                 {
  485.                     ngx_log_error(NGX_LOG_ALERT, cycle->log, 0,
  486.                                   "could not respawn %s",
  487.                                   ngx_processes[i].name);
  488.                     continue;
  489.                 }


  490.                 ch.command = NGX_CMD_OPEN_CHANNEL;
  491.                 ch.pid = ngx_processes[ngx_process_slot].pid;
  492.                 ch.slot = ngx_process_slot;
  493.                 ch.fd = ngx_processes[ngx_process_slot].channel[0];

  494.                 ngx_pass_open_channel(cycle, &ch);

  495.                 live = 1;

  496.                 continue;
  497.             }

  498.             if (ngx_processes[i].pid == ngx_new_binary) {

  499.                 ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx,
  500.                                                        ngx_core_module);

  501.                 if (ngx_rename_file((char *) ccf->oldpid.data,
  502.                                     (char *) ccf->pid.data)
  503.                     == NGX_FILE_ERROR)
  504.                 {
  505.                     ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  506.                                   ngx_rename_file_n " %s back to %s failed "
  507.                                   "after the new binary process \"%s\" exited",
  508.                                   ccf->oldpid.data, ccf->pid.data, ngx_argv[0]);
  509.                 }

  510.                 ngx_new_binary = 0;
  511.                 if (ngx_noaccepting) {
  512.                     ngx_restart = 1;
  513.                     ngx_noaccepting = 0;
  514.                 }
  515.             }

  516.             if (i == ngx_last_process - 1) {
  517.                 ngx_last_process--;

  518.             } else {
  519.                 ngx_processes[i].pid = -1;
  520.             }

  521.         } else if (ngx_processes[i].exiting || !ngx_processes[i].detached) {
  522.             live = 1;
  523.         }
  524.     }

  525.     return live;
  526. }


  527. static void
  528. ngx_master_process_exit(ngx_cycle_t *cycle)
  529. {
  530.     ngx_uint_t  i;

  531.     ngx_delete_pidfile(cycle);

  532.     ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "exit");

  533.     for (i = 0; ngx_modules[i]; i++) {
  534.         if (ngx_modules[i]->exit_master) {
  535.             ngx_modules[i]->exit_master(cycle);
  536.         }
  537.     }

  538.     ngx_close_listening_sockets(cycle);

  539.     /*
  540.      * Copy ngx_cycle->log related data to the special static exit cycle,
  541.      * log, and log file structures enough to allow a signal handler to log.
  542.      * The handler may be called when standard ngx_cycle->log allocated from
  543.      * ngx_cycle->pool is already destroyed.
  544.      */


  545.     ngx_exit_log = *ngx_log_get_file_log(ngx_cycle->log);

  546.     ngx_exit_log_file.fd = ngx_exit_log.file->fd;
  547.     ngx_exit_log.file = &ngx_exit_log_file;
  548.     ngx_exit_log.next = NULL;
  549.     ngx_exit_log.writer = NULL;

  550.     ngx_exit_cycle.log = &ngx_exit_log;
  551.     ngx_exit_cycle.files = ngx_cycle->files;
  552.     ngx_exit_cycle.files_n = ngx_cycle->files_n;
  553.     ngx_cycle = &ngx_exit_cycle;

  554.     ngx_destroy_pool(cycle->pool);

  555.     exit(0);
  556. }


  557. static void
  558. ngx_worker_process_cycle(ngx_cycle_t *cycle, void *data)
  559. {
  560.     ngx_int_t worker = (intptr_t) data;

  561.     ngx_uint_t         i;
  562.     ngx_connection_t  *c;

  563.     ngx_process = NGX_PROCESS_WORKER;

  564.     ngx_worker_process_init(cycle, worker);

  565.     ngx_setproctitle("worker process");

  566. #if (NGX_THREADS)
  567.     {
  568.     ngx_int_t         n;
  569.     ngx_err_t         err;
  570.     ngx_core_conf_t  *ccf;

  571.     ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);

  572.     if (ngx_threads_n) {
  573.         if (ngx_init_threads(ngx_threads_n, ccf->thread_stack_size, cycle)
  574.             == NGX_ERROR)
  575.         {
  576.             /* fatal */
  577.             exit(2);
  578.         }

  579.         err = ngx_thread_key_create(&ngx_core_tls_key);
  580.         if (err != 0) {
  581.             ngx_log_error(NGX_LOG_ALERT, cycle->log, err,
  582.                           ngx_thread_key_create_n " failed");
  583.             /* fatal */
  584.             exit(2);
  585.         }

  586.         for (n = 0; n < ngx_threads_n; n++) {

  587.             ngx_threads[n].cv = ngx_cond_init(cycle->log);

  588.             if (ngx_threads[n].cv == NULL) {
  589.                 /* fatal */
  590.                 exit(2);
  591.             }

  592.             if (ngx_create_thread((ngx_tid_t *) &ngx_threads[n].tid,
  593.                                   ngx_worker_thread_cycle,
  594.                                   (void *) &ngx_threads[n], cycle->log)
  595.                 != 0)
  596.             {
  597.                 /* fatal */
  598.                 exit(2);
  599.             }
  600.         }
  601.     }
  602.     }
  603. #endif

  604.     for ( ;; ) {

  605.         if (ngx_exiting) {

  606.             c = cycle->connections;

  607.             for (i = 0; i < cycle->connection_n; i++) {

  608.                 /* THREAD: lock */

  609.                 if (c[i].fd != -1 && c[i].idle) {
  610.                     c[i].close = 1;
  611.                     c[i].read->handler(c[i].read);
  612.                 }
  613.             }

  614.             ngx_event_cancel_timers();

  615.             if (ngx_event_timer_rbtree.root == ngx_event_timer_rbtree.sentinel)
  616.             {
  617.                 ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "exiting");

  618.                 ngx_worker_process_exit(cycle);
  619.             }
  620.         }

  621.         ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0, "worker cycle");

  622.         ngx_process_events_and_timers(cycle);

  623.         if (ngx_terminate) {
  624.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "exiting");

  625.             ngx_worker_process_exit(cycle);
  626.         }

  627.         if (ngx_quit) {
  628.             ngx_quit = 0;
  629.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0,
  630.                           "gracefully shutting down");
  631.             ngx_setproctitle("worker process is shutting down");

  632.             if (!ngx_exiting) {
  633.                 ngx_close_listening_sockets(cycle);
  634.                 ngx_exiting = 1;
  635.             }
  636.         }

  637.         if (ngx_reopen) {
  638.             ngx_reopen = 0;
  639.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reopening logs");
  640.             ngx_reopen_files(cycle, -1);
  641.         }
  642.     }
  643. }


  644. static void
  645. ngx_worker_process_init(ngx_cycle_t *cycle, ngx_int_t worker)
  646. {
  647.     sigset_t          set;
  648.     uint64_t          cpu_affinity;
  649.     ngx_int_t         n;
  650.     ngx_uint_t        i;
  651.     struct rlimit     rlmt;
  652.     ngx_core_conf_t  *ccf;
  653.     ngx_listening_t  *ls;

  654.     if (ngx_set_environment(cycle, NULL) == NULL) {
  655.         /* fatal */
  656.         exit(2);
  657.     }

  658.     ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);

  659.     if (worker >= 0 && ccf->priority != 0) {
  660.         if (setpriority(PRIO_PROCESS, 0, ccf->priority) == -1) {
  661.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  662.                           "setpriority(%d) failed", ccf->priority);
  663.         }
  664.     }

  665.     if (ccf->rlimit_nofile != NGX_CONF_UNSET) {
  666.         rlmt.rlim_cur = (rlim_t) ccf->rlimit_nofile;
  667.         rlmt.rlim_max = (rlim_t) ccf->rlimit_nofile;

  668.         if (setrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
  669.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  670.                           "setrlimit(RLIMIT_NOFILE, %i) failed",
  671.                           ccf->rlimit_nofile);
  672.         }
  673.     }

  674.     if (ccf->rlimit_core != NGX_CONF_UNSET) {
  675.         rlmt.rlim_cur = (rlim_t) ccf->rlimit_core;
  676.         rlmt.rlim_max = (rlim_t) ccf->rlimit_core;

  677.         if (setrlimit(RLIMIT_CORE, &rlmt) == -1) {
  678.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  679.                           "setrlimit(RLIMIT_CORE, %O) failed",
  680.                           ccf->rlimit_core);
  681.         }
  682.     }

  683. #ifdef RLIMIT_SIGPENDING
  684.     if (ccf->rlimit_sigpending != NGX_CONF_UNSET) {
  685.         rlmt.rlim_cur = (rlim_t) ccf->rlimit_sigpending;
  686.         rlmt.rlim_max = (rlim_t) ccf->rlimit_sigpending;

  687.         if (setrlimit(RLIMIT_SIGPENDING, &rlmt) == -1) {
  688.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  689.                           "setrlimit(RLIMIT_SIGPENDING, %i) failed",
  690.                           ccf->rlimit_sigpending);
  691.         }
  692.     }
  693. #endif

  694.     if (geteuid() == 0) {
  695.         if (setgid(ccf->group) == -1) {
  696.             ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
  697.                           "setgid(%d) failed", ccf->group);
  698.             /* fatal */
  699.             exit(2);
  700.         }

  701.         if (initgroups(ccf->username, ccf->group) == -1) {
  702.             ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
  703.                           "initgroups(%s, %d) failed",
  704.                           ccf->username, ccf->group);
  705.         }

  706.         if (setuid(ccf->user) == -1) {
  707.             ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno,
  708.                           "setuid(%d) failed", ccf->user);
  709.             /* fatal */
  710.             exit(2);
  711.         }
  712.     }

  713.     if (worker >= 0) {
  714.         cpu_affinity = ngx_get_cpu_affinity(worker);

  715.         if (cpu_affinity) {
  716.             ngx_setaffinity(cpu_affinity, cycle->log);
  717.         }
  718.     }

  719. #if (NGX_HAVE_PR_SET_DUMPABLE)

  720.     /* allow coredump after setuid() in Linux 2.4.x */

  721.     if (prctl(PR_SET_DUMPABLE, 1, 0, 0, 0) == -1) {
  722.         ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  723.                       "prctl(PR_SET_DUMPABLE) failed");
  724.     }

  725. #endif

  726.     if (ccf->working_directory.len) {
  727.         if (chdir((char *) ccf->working_directory.data) == -1) {
  728.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  729.                           "chdir(\"%s\") failed", ccf->working_directory.data);
  730.             /* fatal */
  731.             exit(2);
  732.         }
  733.     }

  734.     sigemptyset(&set);

  735.     if (sigprocmask(SIG_SETMASK, &set, NULL) == -1) {
  736.         ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  737.                       "sigprocmask() failed");
  738.     }

  739.     srandom((ngx_pid << 16) ^ ngx_time());

  740.     /*
  741.      * disable deleting previous events for the listening sockets because
  742.      * in the worker processes there are no events at all at this point
  743.      */
  744.     ls = cycle->listening.elts;
  745.     for (i = 0; i < cycle->listening.nelts; i++) {
  746.         ls[i].previous = NULL;
  747.     }

  748.     for (i = 0; ngx_modules[i]; i++) {
  749.         if (ngx_modules[i]->init_process) {
  750.             if (ngx_modules[i]->init_process(cycle) == NGX_ERROR) {
  751.                 /* fatal */
  752.                 exit(2);
  753.             }
  754.         }
  755.     }

  756.     for (n = 0; n < ngx_last_process; n++) {

  757.         if (ngx_processes[n].pid == -1) {
  758.             continue;
  759.         }

  760.         if (n == ngx_process_slot) {
  761.             continue;
  762.         }

  763.         if (ngx_processes[n].channel[1] == -1) {
  764.             continue;
  765.         }

  766.         if (close(ngx_processes[n].channel[1]) == -1) {
  767.             ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  768.                           "close() channel failed");
  769.         }
  770.     }

  771.     if (close(ngx_processes[ngx_process_slot].channel[0]) == -1) {
  772.         ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
  773.                       "close() channel failed");
  774.     }

  775. #if 0
  776.     ngx_last_process = 0;
  777. #endif

  778.     if (ngx_add_channel_event(cycle, ngx_channel, NGX_READ_EVENT,
  779.                               ngx_channel_handler)
  780.         == NGX_ERROR)
  781.     {
  782.         /* fatal */
  783.         exit(2);
  784.     }
  785. }


  786. static void
  787. ngx_worker_process_exit(ngx_cycle_t *cycle)
  788. {
  789.     ngx_uint_t         i;
  790.     ngx_connection_t  *c;

  791. #if (NGX_THREADS)
  792.     ngx_terminate = 1;

  793.     ngx_wakeup_worker_threads(cycle);
  794. #endif

  795.     for (i = 0; ngx_modules[i]; i++) {
  796.         if (ngx_modules[i]->exit_process) {
  797.             ngx_modules[i]->exit_process(cycle);
  798.         }
  799.     }

  800.     if (ngx_exiting) {
  801.         c = cycle->connections;
  802.         for (i = 0; i < cycle->connection_n; i++) {
  803.             if (c[i].fd != -1
  804.                 && c[i].read
  805.                 && !c[i].read->accept
  806.                 && !c[i].read->channel
  807.                 && !c[i].read->resolver)
  808.             {
  809.                 ngx_log_error(NGX_LOG_ALERT, cycle->log, 0,
  810.                               "*%uA open socket #%d left in connection %ui",
  811.                               c[i].number, c[i].fd, i);
  812.                 ngx_debug_quit = 1;
  813.             }
  814.         }

  815.         if (ngx_debug_quit) {
  816.             ngx_log_error(NGX_LOG_ALERT, cycle->log, 0, "aborting");
  817.             ngx_debug_point();
  818.         }
  819.     }

  820.     /*
  821.      * Copy ngx_cycle->log related data to the special static exit cycle,
  822.      * log, and log file structures enough to allow a signal handler to log.
  823.      * The handler may be called when standard ngx_cycle->log allocated from
  824.      * ngx_cycle->pool is already destroyed.
  825.      */

  826.     ngx_exit_log = *ngx_log_get_file_log(ngx_cycle->log);

  827.     ngx_exit_log_file.fd = ngx_exit_log.file->fd;
  828.     ngx_exit_log.file = &ngx_exit_log_file;
  829.     ngx_exit_log.next = NULL;
  830.     ngx_exit_log.writer = NULL;

  831.     ngx_exit_cycle.log = &ngx_exit_log;
  832.     ngx_exit_cycle.files = ngx_cycle->files;
  833.     ngx_exit_cycle.files_n = ngx_cycle->files_n;
  834.     ngx_cycle = &ngx_exit_cycle;

  835.     ngx_destroy_pool(cycle->pool);

  836.     ngx_log_error(NGX_LOG_NOTICE, ngx_cycle->log, 0, "exit");

  837.     exit(0);
  838. }


  839. static void
  840. ngx_channel_handler(ngx_event_t *ev)
  841. {
  842.     ngx_int_t          n;
  843.     ngx_channel_t      ch;
  844.     ngx_connection_t  *c;

  845.     if (ev->timedout) {
  846.         ev->timedout = 0;
  847.         return;
  848.     }

  849.     c = ev->data;

  850.     ngx_log_debug0(NGX_LOG_DEBUG_CORE, ev->log, 0, "channel handler");

  851.     for ( ;; ) {

  852.         n = ngx_read_channel(c->fd, &ch, sizeof(ngx_channel_t), ev->log);

  853.         ngx_log_debug1(NGX_LOG_DEBUG_CORE, ev->log, 0, "channel: %i", n);

  854.         if (n == NGX_ERROR) {

  855.             if (ngx_event_flags & NGX_USE_EPOLL_EVENT) {
  856.                 ngx_del_conn(c, 0);
  857.             }

  858.             ngx_close_connection(c);
  859.             return;
  860.         }

  861.         if (ngx_event_flags & NGX_USE_EVENTPORT_EVENT) {
  862.             if (ngx_add_event(ev, NGX_READ_EVENT, 0) == NGX_ERROR) {
  863.                 return;
  864.             }
  865.         }

  866.         if (n == NGX_AGAIN) {
  867.             return;
  868.         }

  869.         ngx_log_debug1(NGX_LOG_DEBUG_CORE, ev->log, 0,
  870.                        "channel command: %d", ch.command);

  871.         switch (ch.command) {

  872.         case NGX_CMD_QUIT:
  873.             ngx_quit = 1;
  874.             break;

  875.         case NGX_CMD_TERMINATE:
  876.             ngx_terminate = 1;
  877.             break;

  878.         case NGX_CMD_REOPEN:
  879.             ngx_reopen = 1;
  880.             break;

  881.         case NGX_CMD_OPEN_CHANNEL:

  882.             ngx_log_debug3(NGX_LOG_DEBUG_CORE, ev->log, 0,
  883.                            "get channel s:%i pid:%P fd:%d",
  884.                            ch.slot, ch.pid, ch.fd);

  885.             ngx_processes[ch.slot].pid = ch.pid;
  886.             ngx_processes[ch.slot].channel[0] = ch.fd;
  887.             break;

  888.         case NGX_CMD_CLOSE_CHANNEL:

  889.             ngx_log_debug4(NGX_LOG_DEBUG_CORE, ev->log, 0,
  890.                            "close channel s:%i pid:%P our:%P fd:%d",
  891.                            ch.slot, ch.pid, ngx_processes[ch.slot].pid,
  892.                            ngx_processes[ch.slot].channel[0]);

  893.             if (close(ngx_processes[ch.slot].channel[0]) == -1) {
  894.                 ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_errno,
  895.                               "close() channel failed");
  896.             }

  897.             ngx_processes[ch.slot].channel[0] = -1;
  898.             break;
  899.         }
  900.     }
  901. }


  902. #if (NGX_THREADS)

  903. static void
  904. ngx_wakeup_worker_threads(ngx_cycle_t *cycle)
  905. {
  906.     ngx_int_t   i;
  907.     ngx_uint_t  live;

  908.     for ( ;; ) {

  909.         live = 0;

  910.         for (i = 0; i < ngx_threads_n; i++) {
  911.             if (ngx_threads[i].state < NGX_THREAD_EXIT) {
  912.                 if (ngx_cond_signal(ngx_threads[i].cv) == NGX_ERROR) {
  913.                     ngx_threads[i].state = NGX_THREAD_DONE;

  914.                 } else {
  915.                     live = 1;
  916.                 }
  917.             }

  918.             if (ngx_threads[i].state == NGX_THREAD_EXIT) {
  919.                 ngx_thread_join(ngx_threads[i].tid, NULL);
  920.                 ngx_threads[i].state = NGX_THREAD_DONE;
  921.             }
  922.         }

  923.         if (live == 0) {
  924.             ngx_log_debug0(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  925.                            "all worker threads are joined");

  926.             /* STUB */
  927.             ngx_done_events(cycle);

  928.             return;
  929.         }

  930.         ngx_sched_yield();
  931.     }
  932. }


  933. static ngx_thread_value_t
  934. ngx_worker_thread_cycle(void *data)
  935. {
  936.     ngx_thread_t  *thr = data;

  937.     sigset_t          set;
  938.     ngx_err_t         err;
  939.     ngx_core_tls_t   *tls;
  940.     ngx_cycle_t      *cycle;

  941.     cycle = (ngx_cycle_t *) ngx_cycle;

  942.     sigemptyset(&set);
  943.     sigaddset(&set, ngx_signal_value(NGX_RECONFIGURE_SIGNAL));
  944.     sigaddset(&set, ngx_signal_value(NGX_REOPEN_SIGNAL));
  945.     sigaddset(&set, ngx_signal_value(NGX_CHANGEBIN_SIGNAL));

  946.     err = ngx_thread_sigmask(SIG_BLOCK, &set, NULL);
  947.     if (err) {
  948.         ngx_log_error(NGX_LOG_ALERT, cycle->log, err,
  949.                       ngx_thread_sigmask_n " failed");
  950.         return (ngx_thread_value_t) 1;
  951.     }

  952.     ngx_log_debug1(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  953.                    "thread " NGX_TID_T_FMT " started", ngx_thread_self());

  954.     ngx_setthrtitle("worker thread");

  955.     tls = ngx_calloc(sizeof(ngx_core_tls_t), cycle->log);
  956.     if (tls == NULL) {
  957.         return (ngx_thread_value_t) 1;
  958.     }

  959.     err = ngx_thread_set_tls(ngx_core_tls_key, tls);
  960.     if (err != 0) {
  961.         ngx_log_error(NGX_LOG_ALERT, cycle->log, err,
  962.                       ngx_thread_set_tls_n " failed");
  963.         return (ngx_thread_value_t) 1;
  964.     }

  965.     for ( ;; ) {
  966.         thr->state = NGX_THREAD_FREE;

  967. #if 0
  968.         if (ngx_cond_wait(thr->cv, ngx_posted_events_mutex) == NGX_ERROR) {
  969.             return (ngx_thread_value_t) 1;
  970.         }
  971. #endif

  972.         if (ngx_terminate) {
  973.             thr->state = NGX_THREAD_EXIT;

  974.             ngx_log_debug1(NGX_LOG_DEBUG_CORE, cycle->log, 0,
  975.                            "thread " NGX_TID_T_FMT " is done",
  976.                            ngx_thread_self());

  977.             return (ngx_thread_value_t) 0;
  978.         }

  979.         thr->state = NGX_THREAD_BUSY;

  980. #if 0
  981.         if (ngx_event_thread_process_posted(cycle) == NGX_ERROR) {
  982.             return (ngx_thread_value_t) 1;
  983.         }

  984.         if (ngx_event_thread_process_posted(cycle) == NGX_ERROR) {
  985.             return (ngx_thread_value_t) 1;
  986.         }
  987. #endif

  988.         if (ngx_process_changes) {
  989.             if (ngx_process_changes(cycle, 1) == NGX_ERROR) {
  990.                 return (ngx_thread_value_t) 1;
  991.             }
  992.         }
  993.     }
  994. }

  995. #endif


  996. static void
  997. ngx_cache_manager_process_cycle(ngx_cycle_t *cycle, void *data)
  998. {
  999.     ngx_cache_manager_ctx_t *ctx = data;

  1000.     void         *ident[4];
  1001.     ngx_event_t   ev;

  1002.     /*
  1003.      * Set correct process type since closing listening Unix domain socket
  1004.      * in a master process also removes the Unix domain socket file.
  1005.      */
  1006.     ngx_process = NGX_PROCESS_HELPER;

  1007.     ngx_close_listening_sockets(cycle);

  1008.     /* Set a moderate number of connections for a helper process. */
  1009.     cycle->connection_n = 512;

  1010.     ngx_worker_process_init(cycle, -1);

  1011.     ngx_memzero(&ev, sizeof(ngx_event_t));
  1012.     ev.handler = ctx->handler;
  1013.     ev.data = ident;
  1014.     ev.log = cycle->log;
  1015.     ident[3] = (void *) -1;

  1016.     ngx_use_accept_mutex = 0;

  1017.     ngx_setproctitle(ctx->name);

  1018.     ngx_add_timer(&ev, ctx->delay);

  1019.     for ( ;; ) {

  1020.         if (ngx_terminate || ngx_quit) {
  1021.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "exiting");
  1022.             exit(0);
  1023.         }

  1024.         if (ngx_reopen) {
  1025.             ngx_reopen = 0;
  1026.             ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0, "reopening logs");
  1027.             ngx_reopen_files(cycle, -1);
  1028.         }

  1029.         ngx_process_events_and_timers(cycle);
  1030.     }
  1031. }


  1032. static void
  1033. ngx_cache_manager_process_handler(ngx_event_t *ev)
  1034. {
  1035.     time_t        next, n;
  1036.     ngx_uint_t    i;
  1037.     ngx_path_t  **path;

  1038.     next = 60 * 60;

  1039.     path = ngx_cycle->paths.elts;
  1040.     for (i = 0; i < ngx_cycle->paths.nelts; i++) {

  1041.         if (path[i]->manager) {
  1042.             n = path[i]->manager(path[i]->data);

  1043.             next = (n <= next) ? n : next;

  1044.             ngx_time_update();
  1045.         }
  1046.     }

  1047.     if (next == 0) {
  1048.         next = 1;
  1049.     }

  1050.     ngx_add_timer(ev, next * 1000);
  1051. }


  1052. static void
  1053. ngx_cache_loader_process_handler(ngx_event_t *ev)
  1054. {
  1055.     ngx_uint_t     i;
  1056.     ngx_path_t   **path;
  1057.     ngx_cycle_t   *cycle;

  1058.     cycle = (ngx_cycle_t *) ngx_cycle;

  1059.     path = cycle->paths.elts;
  1060.     for (i = 0; i < cycle->paths.nelts; i++) {

  1061.         if (ngx_terminate || ngx_quit) {
  1062.             break;
  1063.         }

  1064.         if (path[i]->loader) {
  1065.             path[i]->loader(path[i]->data);
  1066.             ngx_time_update();
  1067.         }
  1068.     }

  1069.     exit(0);
  1070. }