一般情况下nginx都是采用worker与master的方式启动的,这里我们将分析master进程,我们知道在nginx的main函数的最后会调用ngx_master_process_cycle函数,开始启动worker-master模式,那么我们就从这个函数开始,该函数的定义在src/os/unix/Ngx_process_cycle.c当中:

sigemptyset(&set);
	//向信号集中添加信号
    sigaddset(&set, SIGCHLD);
    sigaddset(&set, SIGALRM);
    sigaddset(&set, SIGIO);
    sigaddset(&set, SIGINT);
    sigaddset(&set, ngx_signal_value(NGX_RECONFIGURE_SIGNAL));
    sigaddset(&set, ngx_signal_value(NGX_REOPEN_SIGNAL));
    sigaddset(&set, ngx_signal_value(NGX_NOACCEPT_SIGNAL));
    sigaddset(&set, ngx_signal_value(NGX_TERMINATE_SIGNAL));
    sigaddset(&set, ngx_signal_value(NGX_SHUTDOWN_SIGNAL));
    sigaddset(&set, ngx_signal_value(NGX_CHANGEBIN_SIGNAL));

//将上面的信号集中的信号全部屏蔽
    if (sigprocmask(SIG_BLOCK, &set, NULL) == -1) {
        ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
                      "sigprocmask() failed");
    }

首先就是直接屏蔽掉一些信号。接下来就是一些比较重要的创建worker进程的部分了:



//获取ngx_core_module模块的配置
    ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);

//开启worker进程,相当于要执行fork了,ccf->worker_processes为需要开启的worker进程的数量
    ngx_start_worker_processes(cycle, ccf->worker_processes,
                               NGX_PROCESS_RESPAWN);
    ngx_start_cache_manager_processes(cycle, 0);

ngx_start_worker_processes函数用来创建worker进程,具体worker进程的数量在ngx_core_module模块的配置的worker_processes域存储。ngx_start_cache_manager_processes用来开启与cache相关的进程。


接下来分析ngx_start_worker_processes函数:

//开启worker进程
static void
ngx_start_worker_processes(ngx_cycle_t *cycle, ngx_int_t n, ngx_int_t type)
{
    ngx_int_t      i;
    ngx_channel_t  ch;   //通过channel传递过去的数据

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

    ch.command = NGX_CMD_OPEN_CHANNEL;    //命令

//这里的n是从配置文件中读取的,表示需要有多少个worker进程,那么就fork出来多少个worker进程
    for (i = 0; i < n; i++) {

        cpu_affinity = ngx_get_cpu_affinity(i);
//在这里创建work进程,ngx_worker_process_cycle为该woker进程的执行函数,执行函数的数据位空,
        ngx_spawn_process(cycle, ngx_worker_process_cycle, NULL,
                          "worker process", type);
//初始化ch的值,ngx_process_slot是刚刚fork出来的进程在ngx_processes数组中的位置
        ch.pid = ngx_processes[ngx_process_slot].pid;
        ch.slot = ngx_process_slot;
//将socketpair的chanel0,也就是写端的描述符传递给其他的进程
        ch.fd = ngx_processes[ngx_process_slot].channel[0];
//向其他的worker传递channel
        ngx_pass_open_channel(cycle, &ch);
    }
}


这里有比较需要注意的地方是,通过一个n循环,调用n次ngx_spawn_process函数创建n个worker进程,在 ngx_spawn_process函数中会将刚刚建立的进程描述符存起来,这里又涉及到一个全局数组ngx_processes(定义在Ngx_process.c文件当中),它用来保存所有worker进程的一些信息,然后会创建一些chanel命令,用于命令以前已经建立的worker进程,让他们跟新socketpair的信息,这是用于进程间通信的,具体的以后再说吧。



这些都执行完了以后,就可以开始进入master进程的主循环了,其实master进程的主循环很是没有意思,说白了就是响应一些信号,调用sigsuspend函数等待信号,这里master进程的信号处理函数在main函数里面都已经初始化完成了,其实他们都是同一个信号处理函数ngx_signal_handler,定义在Ngx_process.c文件当中。该函数会判断接收到的信号,然后设置相应的参数,master进程的主循环就通过这些参数判断接受了哪些信号,然后进行相应的处理。