在通信网络系统中,流量管理的核心是缓存管理、队列管理和调度程序。本文结合使用fpga及ip core阐述缓存管理的结构、工作原理及设计方法
目前硬件高速转发技术的趋势是将整个转发分成两个部分:pe(protocol engine,协议引擎)和tm(traffic management,流量管理)。其中pe完成协议处理,tm负责完成队列调度、缓存管理、流量整形、qos等功能,tm与转发协议无关。
随着通信协议的发展及多样化,协议处理部分pe在硬件转发实现方面,普遍采用现有的商用芯片np(network processor,网络处理器)来完成,流量管理部分需要根据系统的需要进行定制或采用商用芯片来完成。在很多情况下np芯片、tm芯片、交换网芯片无法选用同一家厂商的芯片,这时定制tm成为了成本最低、系统最优化的方案,一般采用fpga来实现,tm的常规结构如图1所示。
图1 tm的常规结构图
目前主流的tm接口均为spi4-p2接口形式,spi4-p2接口信号速率高,tccs(channel-to-channel skew,数据通道的抖动,包含时钟的抖动)难以控制,在常规情况下很难做到很高的速率。spi4-p2接口为达到高速率同时避免tccs问题在很多情况下都对接收端提出了dpa(动态相位调整)的要求。对于spi4-p2接口形式可直接采用altera公司的ip core实现。altera的主流fpga均实现了硬件dpa功能,以stratix ii器件为例,在使能dpa的情况下使用spi4-p2 ip core可实现16gb/s的接口数据速率。
seg模块为数据切分块,根据交换网的数据结构要求,在上交换网的方向上负责把ip包或数据包切分为固定大小的数据块,方便后期的存储调度以及交换网的操作处理,seg模块可配合使用spi4-p2 ip core来实现。与seg模块对应的是rsm模块,rsm模块将从交换网下来的数据块重新组合成完整的ip包或数据包。
bm(buffer management)模块为缓冲管理模块,管理tm的缓冲单元,完成dram的存取操作。外部dram的控制部分可使用使用ddr sdram ip core实现。
qm模块为队列管理模块,负责完成端口的数据队列管理功能,接收bm模块读写dram时的数据入队、出队请求,tm所能支持的数据流的数目、业务类型数目、端口的数目等性能指标在qm模块处体现出来。
scheduler模块为调度模块,根据数据包类型及优先级和端口分配的带宽进行调度,tm流量整形、qos等功能通过调度模块实现。
cell_edit模块完成输出数据的封装,把由dram中读出的数据封装后发送出去。
在tm中需要基于数据服务策略对于不同服务等级的数据包进行不同的管理策略,同时要保证流媒体的数据包不能乱序,数据包有大有小,经过seg模块所分割成的数据块的数目也有多有少,这样就必须有一套行之有效的数据结构基于链表的方法管理这些数据。qm模块基于业务、数据流的方式管理队列,包的管理便由bm模块完成。
bm模块中基于包的数据结构方面由两部分构成:bram和pram。bram为数据缓冲区,对应片外的dram。bram负责存储数据单元,相对于seg模块切分的数据单元,bram内有相应大小的存储单元bcell与之对应,bcell在bram内以地址空间划分,每个bcell相同大小,bcell为bram的最小存取单元。在实际系统中基于seg模块切分的数据单元大小,bcell一般为64~512b。
pram为指针缓冲区,pram对应片外的ssram。pram内部同样以地址空间分为pcell,pcell与bcell一一对应,每一个pcell对应于一个bcell,对应的pcell与bcell地址相同。
pcell的地址对应的代表相应单元的bcell的地址,pcell中的基本信息是下一跳指针。pram与bram关系如图2所示。
图2 pram与bram关系图
在pram中存在两种链表形式,pq list代表已经存储的数据包链表。为方便数据读出,pq list需要记录数据包的第一个数据块地址,即首指针pq_hptr,为方便新的数据写入,pq list需要记录数据包的最后一个数据块地址,即尾指针pq_tptr。pq list同时需要记录该链表的长度作为调度模块进行调度的权值计算使用。
free list代表空闲的地址队列。为方便地辨识、管理空闲的地址,避免地址冲突,在bm中将所有空闲的地址使用一个链表进行管理。这个链表就是空闲地址队列。空闲地址队列依据系统需求的不同有着不同的形式,一般空闲地址队列的构成和pq list相似,由空闲地址首指针free_hptr和空闲地址尾指针free_tptr构成。bm模块的所有操作都围绕着空闲的地址队列free list进行。
基于bm模块的数据流结构,bm模块一般分为write control模块、free list control模块、read control模块、pram control模块、bram control模块。bm的结构如图3所示。
图3 bm结构图
write control模块从free list模块处得到空闲地址,向bram control模块提出写请求,同时更新pram中的内容。free list control模块负责管理空闲地址列表,提供write control模块的写bram地址及pram地址,回收经read control模块读出数据块后释放的地址。read control模块根据调度器的调度结果,通过bram control模块读出需要发送的数据单元,同时将释放的缓冲单元地址写入空闲地址列表。pram control模块为外部ssram的控制模块,可直接使用参考设计完成。bram control模块为外部dram控制模块,一般分为datapath与controler两个子模块。datapath模块专门负责数据接口部分,完成dram接口的dq、dqs处理以及相应的延时调整,controler模块负责完成dram的控制需求。
在bm模块中,bram的带宽与pram的带宽一般为tm的瓶颈。pram的带宽主要受限于访问的次数,而bram的带宽受限于接口带宽。例如对于一个10g的tm,bram的有效带宽必须保证20g,以接口利用率最差只能达到65%计算(考虑seg模块切分信元出现的n+1问题),需要保证接口带宽达到30g。使用64位的dram接口,接口速率不能低于500mb/s,这样对datapath模块的设计提出了更高的要求。在实际系统中,bram主要使用ddr sdram、ddr ii sdram。
当使用stratix ii fpga,bram使用ddr ii sdram时,测试表明ddr ii sdram接口速率可达到800mb/s。在常规使用的情况下,ddr ii sdram接口速率可保证达到667mb/s。对于一个64位的dram接口,接口速率可达到42.7gb/s,完全可以满足一个10g的tm系统。
bm模块作为缓冲管理模块,缓冲的基本单元为bcell,基于对bcell的管理,对于bm的操作都牵涉到空闲地址队列的操作以及链表的操作。最基本的操作就是写入操作和读出操作。bm模块的写入操作由write control模块发起。
对于write control模块,有数据单元需要写入,首先向free list模块申请空闲地址,free list将首指针a给write control模块,作为该数据块的写地址,同时读出首指针a对应在pram中的内容,得到下一跳地址b,将下一跳地址b作为新的空闲地址首指针。
pq list将尾指针n更新为新写入的地址a,同时更新pram中n地址的内容,将a作为下一跳添入n地址。基于节省操作周期,null的内容保留原值,不再更新。这样,一次bram的写入操作需要一次pram的读取操作及一次pram的写入操作。
qm模块接收调度模块的出队信息,将出队的pq链表信息传送给bm模块进行读取操作。
图4 bm模块的写入操作
图5 读出操作的free list堆栈结构
bm模块的读取操作由read control模块发起完成,当有数据单元需要读出,相应的数据单元地址则需要回收进入空闲地址队列free list。对于不同的系统需求,空闲地址队列free list有不同的形式。比较简单的操作是将free list作为堆栈形式使用。
read control模块由pq list的首地址0读出相应的bram中的内容,同时读出pram中对应的下一跳地址1,更新地址1为新的首地址。free list将首指针a更新为刚释放的地址0,同时地址0中写入下一跳指针a。这样一次bram的读出操作需要一次pram的读取操作及一次pram的写入操作。
作为堆栈形式的空闲地址队列在实际操作中会把一部分空闲地址队列放入片内缓冲中。这样在读bram释放地址进入空闲地址队列时可以节省pram的一次写入操作,在写bram时申请空闲地址时可以节省pram的一拍读取操作。pram堆栈结构下内置空闲地址队列表如图6所示。
图6 pram堆栈结构下内置空闲地址队列表
以图5的读出操作为例,当read control模块由pq list的首地址0读出相应的bram中的内容,同时读出pram中对应的下一跳地址1,更新地址1为新的首地址。这时,地址0为已经释放的地址,按空闲队列的操作要求,地址0需要进入空闲地址队列中,在写操作时再将地址0读出提供给write control模块用于写bram。而基于图6的结构,地址0在被释放后不再进行更新pram中的空闲地址队列free list的操作,直接写入片内缓冲中,在write control模块申请地址时由片内缓冲中读出提供给write control模块 。仅在片内free list缓冲几乎满时,进行pram中的空闲地址队列free list的更新操作,或在片内free list缓冲空时进行pram中的空闲地址队列free list的读取操作。
基于图6的结构,在一个读写周期内,可以节省两次pram的操作,在最坏情况下也可节省一次pram的操作。但基于堆栈的结构,栈顶的地址被高频率的反复的调用,栈底的地址很难被使用,dram的工作寿命会因此受到影响。为保证dram的工作寿命,在有些系统中将空闲地址队列free list做成链表形式,从而保证每个dram的存储空间都能被平均的使用。读出操作的free lis链表结构如图7所示。
图7 读出操作的free lis链表结构
read control模块由pq list的首地址0读出相应的bram中的内容,同时读出pram中对应的下一跳地址1,更新地址1为新的首地址。
free list相对于堆栈模式增加尾指针d。free list在回收地址时维持首指针a不变,将尾指针d更新为刚释放的地址0,同时地址d中写入下一跳指针0。这样一次bram的读出操作同样需要一次pram的读取操作及一次pram的写入操作。对于链表方式的空闲地址队列free list,在每个读、写周期必须进行两次pram的写入操作及两次pram的读取操作,pram的效率不高。
针对两种空闲地址队列的效率及对dram的影响,在很多系统中采用了折中的方法,即在pram中使用链表方法管理空闲地址队列free list,在片内采用堆栈模式另建一个空闲地址队列free list,在这种情况下,每个读、写周期需要三次pram的操作。
在实际系统中,bram的带宽与pram的带宽一般为tm的瓶颈,pram主要受限于访问的次数,而bram受限于接口带宽。
在10g的tm系统中,片内数据总线的位宽定为128位,系统时钟定为150mhz,bcell的大小定为64b。在这种情况下,读取操作和写入操作均为4个时钟周期。在满足10g系统的需求下,读取、写入操作周期为7个时钟周期。在前面曾计算过,在满足10g tm系统的情况下,bram采用64位 ddr ii sdram,接口时钟使用250mhz即可满足数据接口的需求。pram采用32位zbt sram ,接口时钟使用系统时钟,每个pcell为64位,每个读、写周期需要6个时钟周期完成。在实际系统中采用altera fpga,bm的设计可以满足10g的tm线速工作的需求。
在40g核心网的tm系统中,片内数据总线的位宽为256位,系统时钟采用250mhz(在40ge的系统中可选用200mhz)。采用ddr ii sdram,接口时钟使用333mhz,则192位的bram可以满足40g的tm需求。此时,bcell可为96b、192b、384b,在这里选用192b。当bcell选用192b时,读取操作和写入操作同样均为6个时钟周期。在满足40g系统的需求下,读取、写入操作周期为9个时钟周期。pram采用48位qdr sram,接口时钟使用150mhz,每个pcell为96位,在每个读、写时钟周期内,pram最多可被操作5次。在采用altera fpga的情况下,bram采用192位 ddr ii sdram,pram采用48位qdr sram,bm的设计可以满足40g的tm线速工作的需求。
|
文章评论(0条评论)
登录后参与讨论