数据包在网络协议栈中从上向下流动时,需要在数据的首部和尾部为其增加相应的包头和包尾。例如在TCP/IP协议栈中,数据从应用层向下传输的过程中,需要为其封装TCP头部、IP头部等,使得接收端能根据各层的报头来正确的接收数据。这就要求存储数据的缓冲区是可变长的,而同时如果数据是通过拷贝在协议层之间传输时,大量的拷贝会极大的影响性能。
因此在VxWorks中,网络协议栈采用了MBlk-ClBlk-Cluster三元组的结构来存储数据。
1)在为数据包封装头部时,申请一个新的mBlk来存放待封装的报头,并将其链接到mBlk链的头部,这样就不需要拷贝了。
2)将真正待传输的上层数据存放在Cluster中,MBlk通过操作Cluster来引用数据。这样数据在协议层之间传输时,只需要修改MBlk的指针即可,不用真正的拷贝数据。
使用三元组的结构来存储网络协议栈中的数据,可以极大的提升系统性能。
1、存储池netPool
存储池netPool与内存池类似,在初始化的时候,就先根据配置构造好一定数量的mBlk、ClBlk和Cluster。然后在需要用到时,从存储池中取出相应的结构,并将其组合在一起即可。
存储池初始化后的示意图如下:
由上图可知,一个存储池又可分为一个mBlk池、一个ClBlk池和多个Cluster池,其中不同的Cluster池中存放着大小不同的Cluster,Cluster的大小默认可以为64、128、256、512、1024、2048这几种。不同Cluster池中Cluster的个数也不相同。
在使用时,根据需要选择大小最合适的Cluster。
netPool的结构体如下所示:
struct netPool /* NET_POOL */
{
M_BLK_ID pmBlkHead; /* head of mBlks */
CL_BLK_ID pClBlkHead; /* head of cluster Blocks */
int mBlkCnt; /* number of mblks */
int mBlkFree; /* number of free mblks */
int clMask; /* cluster availability mask */
int clLg2Max; /* cluster log2 maximum size */
int clSizeMax; /* maximum cluster size */
int clLg2Min; /* cluster log2 minimum size */
int clSizeMin; /* minimum cluster size */
CL_POOL * clTbl [CL_TBL_SIZE]; /* pool table */
M_STAT * pPoolStat; /* pool statistics */
POOL_FUNC * pFuncTbl; /* ptr to function ptr table */
};
每个mBlk、ClBlk、Cluster子池中的元素都通过一个next指针形成链表,而netPool则需要保存链表头部信息即可。
上述结构体中,pmBlkHead指向mBlk池的头部,pClBlkHead指向ClBlk池的头部,clTbl[CL_TBL_SIZE]则保存了不同大小的Cluster形成的各个Cluster池的信息。
这样就可以通过netPool结构来管理mBlk-ClBlk-Cluster三元组结构了。
2、mBlk
mBlk的结构体如下:
typedef struct mBlk
{
M_BLK_HDR mBlkHdr; /* mBlk相关信息 */
M_PKT_HDR mBlkPktHdr; /* 与接收端口相关的信息 */
CL_BLK * pClBlk; /* 指向ClBlk */
} M_BLK;
typedef struct mHdr
{
struct mBlk * mNext; /* next buffer in chain ,形成横向链表*/
struct mBlk * mNextPkt; /* next chain in queue/record ,形成纵向链表*/
char * mData; /* location of data ,指向数据区*/
int mLen; /* amount of data in this mBlk */
UCHAR mType; /* type of data in this mBlk */
UCHAR mFlags; /* flags; see below */
USHORT reserved;
} M_BLK_HDR;
mBlk结构体通过mBlkHdr结构体中的mNext和mNextPkt两个指针,形成纵向和横向两个链表,如下所示:
每一条横向的mBlk链为一个完整的数据包,纵向mBlk链则将多个数据包链接在一起,形成一个数据包链表。
3、clBlk
typedef struct clBlk
{
CL_BLK_LIST clNode; /* union of next clBlk, buffer ptr */
int clSize; /* cluster size */
int clRefCnt; /* reference count of the cluster */
FUNCPTR pClFreeRtn; /* pointer to cluster free routine */
int clFreeArg1; /* free routine arg 1 */
int clFreeArg2; /* free routine arg 2 */
int clFreeArg3; /* free routine arg 3 */
struct netPool * pNetPool; /* pointer to the netPool */
} CL_BLK;
由上可知,clNode是一个联合体,当clBlk空闲时,指向下一个clBlk,即在ClBlk池中形成链表。而当clBlk被使用时,则指向具体的数据区,一般为一个Cluster。
通过clRefCnt完成引用计数,当计数为0时,则将其归还到存储池中。
4、Cluster
当Cluster空闲时,其首部四个字节指向下一个Cluster,从而以链表形式形成一个Cluster池。而当Cluster被使用时,则为具体的数据区。
通过以上的分析可知,一个基本的三元组结构示意图如下:
多个mBlk则通过mBlk.mBlkHdr中的mNext和mNextPkt指针形成横向和纵向的链表,来构造实际的数据包。