`
simohayha
  • 浏览: 1412280 次
  • 性别: 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源代码剖析

    Nginx支持子请求(sub_request),即在一个请求中发起新的子请求,常用于实现模块间的通信或者动态内容的生成。这部分讲解了子请求的创建、执行以及与主请求的交互过程。 这本书全面深入地剖析了Nginx的源代码,...

    Nginx源码剖析

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

    nginx模块开发指南(中文)

    在Nginx中,位置可以继承父位置的配置,因此需要有一个函数来处理这种继承。 #### 2.4 模块定义 模块定义包括了模块的名称、版本以及一系列必要的函数指针,例如处理函数、过滤函数等。 **示例**: ```c static ...

    CentOS VPS如何打开80端口解决网站打不开的问题.docx

    Chain OUTPUT (policy ACCEPT) num target prot opt source destination ``` 如果在"INPUT"链中看到有两条允许80端口的规则,那是因为iptable规则默认是按顺序执行的,重复的规则不会造成问题,但通常只保留一条...

    基于Matlab的三相逆变器SPWM调制仿真及波形分析

    内容概要:本文详细介绍了利用Matlab进行三相逆变器SPWM调制仿真的全过程。首先明确了将110V直流电转换为220V/50Hz交流电的需求,接着阐述了SPWM调制的基本原理及其MATLAB代码实现方法,包括载波频率、调制比的选择以及三相调制信号的生成。文中还展示了逆变桥的Simulink模型搭建步骤,强调了关键参数如死区时间和LC滤波器的设计要点。通过对输出电压、电流波形的分析,验证了模型的有效性和稳定性,并提供了调参技巧和常见问题解决方案。 适合人群:从事电力电子、新能源系统研究的技术人员及高校相关专业师生。 使用场景及目标:适用于希望深入了解SPWM调制机制、掌握三相逆变器设计与优化方法的研究者;可用于教学实验、项目开发和技术培训等场合。 其他说明:文中提供的完整代码和详细的注释有助于初学者快速入门,而丰富的波形分析则为进阶用户提供宝贵的参考资料。此外,作者还分享了一些实用的小贴士,如如何避免常见的错误配置,确保仿真的顺利进行。

    Android5.3 Android 组件-基础控件-例子-broadcast图片

    Android5.3 Android 组件-基础控件-例子-broadcast图片

    基于MATLAB的风光场景生成与削减:概率距离快速削减法的应用

    内容概要:本文详细介绍了如何使用MATLAB实现风光场景生成与削减,特别是在处理风电和光伏发电的不确定性方面。文章首先解释了风光场景生成的概念,即通过蒙特卡洛方法模拟各种可能的风电和光伏出力情况。接着,针对生成的大量场景可能导致计算复杂的问题,提出了基于概率距离的快速削减法,通过计算场景间的概率距离,将大量场景削减至少数关键场景,从而简化后续计算。文中还展示了具体的MATLAB代码实现,包括场景生成、距离计算、场景削减等关键步骤,并讨论了一些优化技巧如并行计算、数据标准化等。此外,文章强调了该方法在电力系统调度中的重要性和实用性,能够有效提高计算效率和准确性。 适合人群:从事电力系统研究的科研人员、工程师及相关领域的学生。 使用场景及目标:适用于需要处理风电、光伏等可再生能源出力不确定性的电力系统调度和优化项目。主要目标是通过减少计算复杂度,提高调度模型的计算效率和准确性。 其他说明:文中提供的MATLAB代码具有良好的可移植性和复用性,可以根据具体应用场景进行适当修改和扩展。

    毕业设计(论文) 直动式液压往复泵设计说明书.doc.doc

    毕业设计(论文) 直动式液压往复泵设计说明书.doc.doc

    电动剃须刀的结构设计说明书.docx.docx

    电动剃须刀的结构设计说明书.docx.docx

    【深度学习应用】DeepSeek智能模型的万能提问模板设计:提升问题解决效率与准确性

    内容概要:本文介绍了DeepSeek这一开源深度学习模型的使用方法和提问技巧。DeepSeek不仅能够执行指令,还能进行思考和推理。文章提出了几种提问模板,帮助用户更有效地利用DeepSeek解决问题。其中包括“背景+需求+约束条件”、“身份+任务+要求+例子”、“我要做什么→要做什么用→希望达到什么效果→但担心什么问题”、“需求+担忧+反向验证”、“问题+追问预期+调整方向”以及“目标+条件+验证方式”。每种模板都配有详细的解释和示例,以指导用户如何构建问题,从而获得更有针对性的答案。 适合人群:适合对深度学习感兴趣的开发者、研究人员以及希望通过DeepSeek解决实际问题的用户。 使用场景及目标:①帮助用户更好地理解和使用DeepSeek,提高问题解决效率;②通过具体的提问模板,让用户能够更精准地获取所需信息;③适用于各种场景,包括但不限于技术问题解答、学习计划制定、课程推广方案设计等。 其他说明:为了充分利用DeepSeek的功能,用户在提问时应尽量提供详尽的背景信息,明确需求,并根据实际情况设置合理的约束条件。此外,用户还可以通过设定角色和任务,让DeepSeek从特定角度进行思考,从而得到更具创意和实用性的解决方案。

    时间序列预测中SVM、BP神经网络与LSTM的应用及MATLAB实现

    内容概要:本文详细介绍了支持向量机(SVM)、BP神经网络和LSTM网络在时间序列预测中的应用及其MATLAB实现。首先,针对SVM部分,文章讲解了如何将时间序列数据转化为特征矩阵并通过滑动窗口进行预测,强调了交叉验证的重要性。对于BP神经网络,则着重于数据归一化以及网络结构的选择,并指出BP网络容易陷入局部最优的问题。最后,在LSTM部分,文章展示了如何利用MATLAB的深度学习工具箱实现LSTM网络,解决了输入数据格式转换的问题,并实现了滚动预测。文中提供了具体的代码示例,确保读者能够轻松替换数据源并运行程序。 适用人群:适用于有一定MATLAB基础的时间序列预测初学者和技术爱好者。 使用场景及目标:帮助读者掌握三种不同类型的机器学习模型在时间序列预测任务中的具体应用,理解各自的特点和优缺点,从而选择合适的模型应用于实际项目中。 其他说明:文章不仅提供了详细的理论解释,还给出了完整的代码实现,使得读者可以直接上手实践。同时,作者提醒了一些常见的错误和注意事项,如数据格式、归一化等问题,有助于提高预测准确性。

    (精品)热转印制作人物主题个性杯子毕业论文设计说明书.doc

    (精品)热转印制作人物主题个性杯子毕业论文设计说明书.doc

    5000吨年氧化羰化制碳酸二甲酯合成工艺设计说明书.doc

    5000吨年氧化羰化制碳酸二甲酯合成工艺设计说明书.doc

    3D文件可爱小鹿3D建模,可爱小鹿建模仿真,小鹿仿真设计,3D打印文件,免费下载

    【3D文件】可爱小鹿3D建模,3D打印文件,免费下载 资源图片查看链接:https://blog.csdn.net/weixin_60324241/article/details/147233616?spm=1011.2415.3001.5331 欢迎查看下载

    基于Simulink的BUCK变换器与BLDCM控制系统设计及优化

    内容概要:本文详细介绍了如何利用Simulink进行BUCK变换器的电压电流双闭环控制以及多相控制的设计方法,并探讨了无刷直流电机(BLDCM)控制系统的构建。文中不仅涵盖了基本的控制理论,还提供了具体的MATLAB/Simulink代码示例,帮助读者理解和实现复杂的电力电子控制系统。对于常见的调试问题,如代数环错误、谐波失真、电流不平衡等,作者分享了许多实践经验和技术技巧。 适合人群:从事电力电子、自动化控制领域的工程师和技术人员,尤其适用于有一定Simulink基础并希望深入掌握BUCK变换器和BLDCM控制设计的专业人士。 使用场景及目标:① 学习如何在Simulink中搭建和优化BUCK变换器的电压电流双闭环控制系统;② 掌握多相BUCK变换器的均流控制策略;③ 实现BLDCM的高效换相逻辑和速度环参数整定;④ 解决仿真过程中可能出现的各种问题。 其他说明:文章强调了实际应用中的注意事项,如参数选择、模块配置、信号处理等方面的经验,有助于提高仿真的准确性和效率。同时提醒读者关注硬件保护措施,避免实验过程中损坏设备。

    基于粒子群算法的电力系统经济调度模型研究 - IEEE30节点六机系统的优化实现

    内容概要:本文详细介绍了如何利用粒子群优化(PSO)算法解决电力系统经济调度问题,特别是在IEEE30节点六机系统中的应用。文章首先解释了关键参数的选择及其意义,如最大迭代次数、惯性权重、学习因子等。接着深入探讨了适应度函数的设计,包括火电成本计算、失负荷惩罚以及弃风弃光惩罚。此外,还讨论了重要的约束处理方法,如爬坡速率限制和出力上下限截断。通过这些措施,实现了对六个发电机组的智能调度,使得总发电成本最小化,同时保证了系统的稳定性和可靠性。实验结果显示,相比传统调度方案,新方法使总成本降低了12.7%,弃风弃光量控制在5%以内。 适合人群:从事电力系统调度、优化算法研究的专业人士,尤其是对智能算法应用于能源管理感兴趣的工程师和技术研究人员。 使用场景及目标:适用于希望提高电力系统经济效益和环境效益的企业和个人。主要目标是在满足电力需求的前提下,减少化石燃料消耗,增加可再生能源利用率,从而达到节能减排的目的。 其他说明:文中提供的代码片段仅为示意,具体实现时还需考虑更多实际情况,如不同类型的发电机组特性、精确的气象预测数据等。此外,尽管PSO算法表现良好,但在某些极端情况下仍可能存在局限性,因此建议结合其他优化手段共同使用。

    基于MATLAB的楼宇负荷需求响应研究:模型预测控制与完全复现

    内容概要:本文详细介绍了如何使用MATLAB和CVX工具箱,通过模型预测控制(MPC)优化楼宇空调系统的能耗管理。首先,作者通过对建筑热力学特性的建模,建立了状态空间模型来描述楼宇的温度变化规律。接着,设计了一个24小时预测时域的MPC控制器,结合动态电价信号,实现了在电价低谷期预冷建筑并在电价高峰期减少制冷功率的目标。文中展示了具体的MATLAB代码片段,解释了关键参数的选择及其物理意义,并讨论了如何通过引入舒适度约束和反馈校正机制提高控制精度。最终,仿真实验表明该方法不仅能够显著降低电费支出,还能保持室内温度在舒适的范围内。 适合人群:从事楼宇自动化、能源管理和控制系统设计的研究人员和技术人员。 使用场景及目标:适用于商业建筑和工业园区的空调系统节能改造项目,旨在通过智能化控制手段实现节能减排和成本优化。 其他说明:文章强调了模型简化与计算效率之间的权衡,以及在实际应用中需要根据不同建筑物的具体情况进行参数调整。同时指出,良好的用户体验是衡量控制算法成功与否的重要标准之一。

    基于CANoe的Bootloader上位机软件开发:飞思卡尔MC9S12G128MLL的应用与优化

    内容概要:本文详细介绍了基于CANoe的Bootloader上位机软件开发流程及其应用,重点在于飞思卡尔MC9S12G128MLL这款16位单片机。首先解释了选择CANoe的原因,因其强大的CAN总线仿真能力和灵活的数据格式定义,非常适合Bootloader开发。接着深入探讨了下位机MC9S12G128MLL的基本配置,包括系统时钟、端口初始化以及CAN模块的设置。随后展示了具体的CAN报文收发实现方法,如通过CAPL语言编写上位机发送命令,下位机接收并处理这些命令,从而实现进入Bootloader模式的功能。此外还讨论了进一步的功能扩展,如程序下载、校验等复杂操作的设计思路和技术细节。最后强调了实际开发过程中遇到的问题及解决方案,例如Flash擦写、通信协议设计、CRC校验等方面的经验教训。 适合人群:从事嵌入式系统开发的技术人员,尤其是专注于汽车电子领域CAN总线相关项目的工程师。 使用场景及目标:适用于需要开发或优化基于CANoe的Bootloader上位机软件的项目,旨在提高嵌入式系统的升级效率和可靠性,确保在复杂环境下能够顺利完成固件更新任务。 其他说明:文中不仅提供了详细的理论讲解,还包括大量实用的代码示例,帮助读者更好地理解和掌握具体实施步骤。同时,作者分享了许多宝贵的实践经验,有助于避免常见错误,提升开发成功率。

    dwaDWADWAWADWAWDADWADWA

    dwaDWADWAWADWAWDADWADWA

    基于S7-300 PLC与组态王的散装水泥生产线在线称重控制系统设计与实现

    内容概要:本文详细介绍了利用西门子S7-300 PLC和组态王软件构建散装水泥生产线在线称重控制系统的全过程。首先阐述了硬件配置,包括CPU315-2DP作为主站、ET200M分布式从站以及SM331模拟量模块用于接收称重传感器信号。接着深入解析了梯形图程序的关键部分,如OB35中断组织块用于每100ms采集一次重量数据,并采用浮点数比较器实时判断重量变化趋势。此外,还讨论了组态王的画面设计,包括实时重量显示、趋势图控件的应用以及报警机制的设置。文中特别强调了针对动态称重过程中信号跳变的问题采取了移动平均滤波算法进行优化,并分享了一些实际工程中的经验和教训,如正确的接线方式、抗干扰措施等。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是那些对PLC编程、组态软件应用有一定基础的人群。 使用场景及目标:适用于新建或改造散装水泥生产线的称重控制系统项目。主要目标是提高称重精度,确保生产效率的同时保障产品质量。通过学习本文可以掌握完整的系统设计方案,包括硬件选型、软件编程及人机界面设计等方面的知识。 其他说明:文中提供了大量具体的代码片段和配置细节,有助于读者更好地理解和实施相关技术。同时,作者结合自身经验给出了许多宝贵的建议,对于初学者来说非常有价值。

Global site tag (gtag.js) - Google Analytics