您的位置:首页 > 运维架构 > Nginx

Nginx源码分析-master和worker进程间的通信

2013-03-06 13:55 537 查看
转自 http://blog.csdn.net/marcky/article/details/6014733

前面单独分析了master进程和worker的工作情况,本文就大概看一下master进程和worker进程之间是如何使用channel来完成通信的。这部分实现的源码主要分布于src/os/unix/channel.h和channel.c两个文件中。实现极其简单,没有什么复杂的逻辑。下面,我绘制了一个简单的master进程和worker进程间的关系,图中的箭头符号指出数据是由master进程传给worker进程,而没有从worker到master;这是因为channel不是一个普通的数据传输管道,在Nginx中它仅仅是用着master发送指令给worker的一个管道,master借此channel来告诉worker进程该做什么了,worker却不需要告诉master该做什么,所以是一个单向的通道。



master进程每次发送给worker进程的指令用如下一个结构来完成封装:

[cpp] view plaincopyprint?

typedef struct {

ngx_uint_t command;

ngx_pid_t pid;

ngx_int_t slot;

ngx_fd_t fd;

} ngx_channel_t;

[cpp] view plaincopyprint?

ch.command = NGX_CMD_CLOSE_CHANNEL;

ch.fd = -1;

ch.pid = ngx_processes[i].pid;

ch.slot = i;

ngx_write_channel(ngx_processes
.channel[0],

&ch, sizeof(ngx_channel_t), cycle->log);

ch.command = NGX_CMD_CLOSE_CHANNEL;
ch.fd = -1;
ch.pid = ngx_processes[i].pid;
ch.slot = i;
ngx_write_channel(ngx_processes
.channel[0],
&ch, sizeof(ngx_channel_t), cycle->log);


这几行代码是我从ngx_reap_children函数中拼凑起来的,所以看上去好像有点奇怪,不那么顺畅;但却清晰的给我们展现了master进程怎么给一个worker进程发送指令,此处发送的指令时NGX_CMD_CLOSE_CHANNEL。发送指令的函数ngx_write_channel是利用sendmsg来完成,《Unix网络编程》可以详细了解sendmsg。

worker进程在调用ngx_worker_process_init进行初始化的时候,使用了如下两行代码将channel放到epoll等事件处理模块中。

[cpp] view plaincopyprint?

if (ngx_add_channel_event(cycle, ngx_channel, NGX_READ_EVENT,

ngx_channel_handler)

== NGX_ERROR)

{

/* fatal */

exit(2);

}

[cpp] view plaincopyprint?

/*

读出master进程发送给过来的指令数据, ngx_read_channel

是利用recvmsg实现,详细介绍见《unix网络编程》

*/

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

/*

判断command的值,从而采取具体的动作,代码意图都写得很明显,

就不在这里多说了。

*/

switch (ch.command) {

case NGX_CMD_QUIT:

ngx_quit = 1;

break;

case NGX_CMD_TERMINATE:

ngx_terminate = 1;

break;

case NGX_CMD_REOPEN:

ngx_reopen = 1;

break;

case NGX_CMD_OPEN_CHANNEL:

ngx_processes[ch.slot].pid = ch.pid;

ngx_processes[ch.slot].channel[0] = ch.fd;

break;

case NGX_CMD_CLOSE_CHANNEL:

if (close(ngx_processes[ch.slot].channel[0]) == -1) {

ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_errno,

"close() channel failed");

}

ngx_processes[ch.slot].channel[0] = -1;

break;

}

/*
读出master进程发送给过来的指令数据, ngx_read_channel
是利用recvmsg实现,详细介绍见《unix网络编程》
*/
n = ngx_read_channel(c->fd, &ch, sizeof(ngx_channel_t), ev->log);
/*
判断command的值,从而采取具体的动作,代码意图都写得很明显,
就不在这里多说了。
*/
switch (ch.command) {
case NGX_CMD_QUIT:
ngx_quit = 1;
break;
case NGX_CMD_TERMINATE:
ngx_terminate = 1;
break;
case NGX_CMD_REOPEN:
ngx_reopen = 1;
break;
case NGX_CMD_OPEN_CHANNEL:
ngx_processes[ch.slot].pid = ch.pid;
ngx_processes[ch.slot].channel[0] = ch.fd;
break;
case NGX_CMD_CLOSE_CHANNEL:
if (close(ngx_processes[ch.slot].channel[0]) == -1) {
ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_errno,
"close() channel failed");
}
ngx_processes[ch.slot].channel[0] = -1;
break;
}


Nginx中关于整个channel的实现就这么简单,没有什么多余的事情。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: