`
simohayha
  • 浏览: 1400071 次
  • 性别: Icon_minigender_1
  • 来自: 火星
社区版块
存档分类
最新评论

nginx中的output chain的处理(一)

阅读更多
这里我们详细来看ngx_linux_sendfile_chain方法,这个函数也就是nginx的发送函数。

一般来说,我们最终都会调用这个函数来发送最终的数据,因此我们来着重分析这个函数,这里主要就是对buf的一些参数的理解。

来看函数原型:
ngx_chain_t *
ngx_linux_sendfile_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)

第一个参数是当前的连接,第二个参数是所需要发送的chain,第三个参数是所能发送的最大值。

然后来看这里的几个重要的变量:


send 表示将要发送的buf已经已经发送的大小。
sent表示已经发送的buf的大小
prev_send 表示上一次发送的大小,也就是已经发送的buf的大小。
fprev 和prev-send类似,只不过是file类型的。

complete表示是否buf被完全发送了,也就是sent是否等于send - prev_send.

header表示需要是用writev来发送的buf。也就是only in memory的buf。

struct iovec  *iov, headers[NGX_HEADERS] 这个主要是用于sendfile和writev的参数,这里注意上面header数组保存的就是iovec。



然后我们来看初始化

   
wev = c->write;

    if (!wev->ready) {
        return in;
    }

    if (limit == 0 || limit > (off_t) (NGX_SENDFILE_LIMIT - ngx_pagesize)) {
        limit = NGX_SENDFILE_LIMIT - ngx_pagesize;
    }


    send = 0;
//设置header,也就是in memory的数组
    header.elts = headers;
    header.size = sizeof(struct iovec);
    header.nalloc = NGX_HEADERS;
    header.pool = c->pool;


这里nginx的处理核心思想就是合并内存连续并相邻的buf(不管是in memory还是in file)

下面这段代码就是处理in memory的部分,然后将buf放入对应的iovec数组。


//开始遍历
for (cl = in;
             cl && header.nelts < IOV_MAX && send < limit;
             cl = cl->next)
        {
            if (ngx_buf_special(cl->buf)) {
                continue;
            }
//如果不止是在buf中,这是因为有时in file的文件我们可能需要内存中也有拷贝,所以如果一个buf同时in memoey和in file的话,nginx会认为它是in file的来处理。
            if (!ngx_buf_in_memory_only(cl->buf)) {
                break;
            }

//得到buf的大小
            size = cl->buf->last - cl->buf->pos;

//大于limit的话修改为size
            if (send + size > limit) {
                size = limit - send;
            }
//如果prev等于pos,则说明当前的buf的数据和前一个buf的数据是连续的。
            if (prev == cl->buf->pos) {
                iov->iov_len += (size_t) size;

            } else {
//否则说明是不同的buf,因此add一个iovc。
                iov = ngx_array_push(&header);
                if (iov == NULL) {
                    return NGX_CHAIN_ERROR;
                }

                iov->iov_base = (void *) cl->buf->pos;
                iov->iov_len = (size_t) size;
            }

//这里可以看到prev保存了当前buf的结尾。
            prev = cl->buf->pos + (size_t) size;
//更新发送的大小
            send += size;
        }


然后是in file的处理这里比较核心的一个判断就是fprev == cl->buf->file_pos,和上面的in memory类似,fprev保存的就是上一次处理的buf的尾部。这里如果这两个相等,那就说明当前的两个buf是连续的(文件连续).

ok.来看代码。


//可以看到如果header的大小不为0则说明前面有需要发送的buf,因此我们就跳过in file处理
if (header.nelts == 0 && cl && cl->buf->in_file && send < limit) {
//得到file
            file = cl->buf;

//开始合并。
            do {
//得到大小
                size = cl->buf->file_last - cl->buf->file_pos;

//如果太大则进行对齐处理。
                if (send + size > limit) {
                    size = limit - send;

                    aligned = (cl->buf->file_pos + size + ngx_pagesize - 1)
                               & ~((off_t) ngx_pagesize - 1);

                    if (aligned <= cl->buf->file_last) {
                        size = aligned - cl->buf->file_pos;
                    }
                }

//设置file_size.
                file_size += (size_t) size;
//设置需要发送的大小
                send += size;
//和上面的in memory处理一样就是保存这次的last
                fprev = cl->buf->file_pos + size;
                cl = cl->next;

            } while (cl
                     && cl->buf->in_file
                     && send < limit
                     && file->file->fd == cl->buf->file->fd
                     && fprev == cl->buf->file_pos);
        }


然后就是发送部分,这里in file使用sendfile,in memory使用writev.这里处理比较简单,就是发送然后判断发送的大小

if (file) {
#if 1
            if (file_size == 0) {
                ngx_debug_point();
                return NGX_CHAIN_ERROR;
            }
#endif
#if (NGX_HAVE_SENDFILE64)
            offset = file->file_pos;
#else
            offset = (int32_t) file->file_pos;
#endif

//发送数据
            rc = sendfile(c->fd, file->file->fd, &offset, file_size);
......................................................
//得到发送的字节数
            sent = rc > 0 ? rc : 0;

        } else {
            rc = writev(c->fd, header.elts, header.nelts);
.......................................................................

            sent = rc > 0 ? rc : 0;
}
        


接下来这部分就是更新标记的部分,主要是buf的标记。

这里要注意一个地方,那就是ngx_buf_size部分,这个宏很简单就是判断buf是不是在memory中,如果是的话,就用pos和last计算,否则认为是在file中。

可是这里就有个问题了,如果一个buf本来是在file中的,我们由于某种原因,在内存中也有一份拷贝,可是我们并没有修改内存中的副本,于是如果我们还需要切割这个buf,这个时候,如果last和pos也就是buf对应的指针没有设置正确的话,这里就会出现问题了。

这里我觉得应该还有个标记,那就是如果内存中的副本我只是只读的话,发送的时候不应该算它在memory中。


//如果send - prev_send == sent则说明该发送的都发完了。
if (send - prev_send == sent) {
            complete = 1;
        }
//更新congnect的sent域。
        c->sent += sent;

//开始重新遍历chain,这里是为了防止没有发送完全的情况,此时我们就需要切割buf了。
        for (cl = in; cl; cl = cl->next) {

            if (ngx_buf_special(cl->buf)) {
                continue;
            }

            if (sent == 0) {
                break;
            }
//得到buf size
            size = ngx_buf_size(cl->buf);

//如果大于当前的size,则说明这个buf的数据已经被完全发送完毕了。,因此更新它的域。
            if (sent >= size){
//更新sent域
                sent -= size;
//如果在内存则更新pos
                if (ngx_buf_in_memory(cl->buf)) {
                    cl->buf->pos = cl->buf->last;
                }
//如果在file
                if (cl->buf->in_file) {
                    cl->buf->file_pos = cl->buf->file_last;
                }

                continue;
            }

//到这里说明当前的buf只有一部分被发送出去了,因此这里我们只需要修改指针。以便于下次发送。
            if (ngx_buf_in_memory(cl->buf)) {
                cl->buf->pos += (size_t) sent;
            }
//同上。
            if (cl->buf->in_file) {
                cl->buf->file_pos += sent;
            }

            break;
        }


最后一部分就是一些是否退出循环的操作。这里要注意,nginx中如果发送未完全的话,将会直接返回的,返回的就是没有发送完毕的chain,它的buf也已经被更新。这是因为nginx是单线程的,不能有任何意义的空跑和阻塞,因此当complete为0,nginx就认为是系统负载过大,此时直接返回,然后处理其他的事情,等待和下次的chain一起发送。


if (eintr) {
            continue;
        }
//如果未完成,则返回。
        if (!complete) {
            wev->ready = 0;
            return cl;
        }

        if (send >= limit || cl == NULL) {
            return cl;
        }
//更新in,也就是开始处理下一个chain
        in = cl;






1
1
分享到:
评论
1 楼 liuxuejin 2013-09-13  
为什么一直都说 sendfile 在linux 2.6 中移除了???

相关推荐

    Nginx中http请求处理过程

    phases 是 Nginx 中的一个数组,用于存储 HTTP 请求处理过程中的各个阶段。在初始化时,Nginx 会将 phases 数组中的每个元素设置为默认值。 2.2.1 理论说明: phases 数组的初始化是 Nginx 启动过程中的一个关键...

    补充:Nginx之模块处理流程

    如果多个处理模块映射到了同一个位置,Nginx会在配置文件中处理冲突,确保只有一个模块处理请求。 3. **过滤模块(Filter Modules)**:过滤模块在处理模块之后介入,用于修改或增强处理模块生成的输出。例如,它们...

    Nginx 中文官方手册.CHM

    Nginx 中文官方手册.CHM

    第一个Nginx模块的例子

    标题中的“第一个Nginx模块的例子”意味着我们将探讨如何创建一个自定义的Nginx模块。Nginx是一个高性能的Web服务器和反向代理服务器,它以其轻量级、高并发处理能力而闻名。开发自定义模块可以让用户扩展Nginx的...

    详解nginx请求头数据读取流程

    在这个过程中,Nginx会检查每个请求头,将其存储到内存中,并执行相应的处理。 1. 缓冲区管理:当缓冲区中没有足够的空间存储新请求头时,`ngx_http_process_request_headers()`会分配更大的缓冲区。如果客户端发送...

    Nginx配置文件——一级域名、二级域名

    在IT行业中,Nginx是一款广泛应用的高性能HTTP和反向代理服务器,它的配置灵活性和高效性使其成为搭建网站和处理高并发请求的理想选择。本文将深入探讨如何通过Nginx配置文件来设置一级域名和二级域名,以及如何利用...

    Nginx源码剖析

    - **Handler**: Handler 是 Nginx 中处理特定类型请求的模块,例如静态文件服务、动态内容代理等。 - **Filter**: Filter 是一种特殊的 handler,用于对响应进行修改或增强。 #### 1.4 Nginx 的 filter 的处理 ...

    升级gitlab中nginx版本.docx

    在实际生产环境中,GitLab系统的Nginx版本升级和配置是一个非常重要的任务。为确保系统的稳定性和安全性,需要对GitLab系统中的Nginx版本进行升级和配置。本文将详细介绍如何升级GitLab系统中的Nginx版本,使其使用...

    nginx lua处理图片

    2. **配置Nginx**:在`nginx.conf`文件中,你需要设置一个location块来处理图片请求。在这个location块内,可以使用`content_by_lua_file`指令来指定一个Lua脚本文件,比如`thumbnail.lua`或`config.lua`,这样Nginx...

    nginx rtmp转发服务器

    Nginx RTMP服务器是一个基于Nginx的开源扩展,专门设计用于处理实时流媒体协议(RTMP)。它为内容发布者和消费者提供了一个高效、低延迟的平台,适用于直播、视频点播等多种应用场景。在本文中,我们将深入探讨Nginx...

    nginx官方文档中文版

    Nginx 是一个开源的 Web 服务器软件,可以作为 Web 服务器、反向代理服务器、缓存服务器、负载均衡器和媒体流服务器等。下面是 Nginx 官方文档中文版的知识点总结: 安装和配置 * 从源码构建 Nginx * Nginx 初学者...

    在Android app中集成nginx,非JNI

    在Android环境中,nginx可以帮助我们处理HTTP请求,服务于本地应用的静态资源,或者在移动设备上搭建一个本地测试环境。 要将nginx集成到Android应用中,我们需要以下步骤: 1. **获取nginx源码**:首先,从nginx...

    NGINX实现一个域名访问多个项目1

    标题中的"NGINX实现一个域名访问多个项目1"是指利用Nginx服务器的配置能力,让同一个域名能够根据不同的URL路径指向不同的应用或项目。描述中提到,这是为了解决在一个域名下部署多个项目的问题,避免为每个项目单独...

    nginx1.18 nginx1.18 nginx1.18

    Nginx 是一个流行的开源 Web 服务器,以其高性能、高并发处理能力而闻名,常用于静态内容服务和反向代理。在这个版本中,Nginx 提供了稳定性和性能上的优化,以及可能的新功能和安全更新。 首先,让我们深入了解...

    nginx HTTP处理流程.docx

    Nginx是一款高性能的Web服务器和反向代理服务器,由伊戈尔·赛索耶夫(Igor Sysoev)开发,以其高效的并发处理能力和低内存消耗而闻名。它广泛应用于各大互联网公司,如百度、京东、新浪、网易、腾讯、淘宝等。Nginx的...

    详解nginx同一端口监听多个域名和同时监听http与https

    具体操作方法是在Nginx的配置文件中设置多个server块,每个server块的server_name指令指定一个域名。当有请求到达时,Nginx会根据HTTP请求头中的Host字段来决定将请求转发到哪一个server块。这里有一个重要的注意...

    nginx流媒体安装包(nginx_mod_h264_streaming,yamdi)

    这会生成一个名为output.m3u8的主播放列表文件和一系列.ts分片文件,它们可以被Nginx的流媒体模块识别和处理。 三、整合与测试 将转换后的视频文件上传到Nginx配置的`hls_path`目录,然后可以通过浏览器访问`...

    nginx+中文文档

    4. **静态文件处理**:Nginx 对于静态文件的处理非常高效,能快速响应请求,减少对后端动态服务器的压力。 5. **缓存**:Nginx 提供了HTTP缓存功能,可以缓存常用的静态资源,提高访问速度。 ### Nginx 安装过程 ...

    nginx中文文档

    此外,根据Netcraft统计,2012年8月份,世界上最繁忙的网站中有超过十分之一使用Nginx作为服务器或代理服务器。成功的案例包括Netflix、***和FastMail.FM等。 Nginx具备多种服务器特性,包括处理静态文件、索引文件...

Global site tag (gtag.js) - Google Analytics