网络协议分析课程设计之协议编程
实验一帧封装
实验目的:
•编写程序,根据给出的原始数据,组装一个IEEE
802.3格式的帧(题目)默认的输入文件为二进制原始数据(文件名分别为input1和input 2))。
•要求程序为命令行程序。比如,可执行文件名为framer.exe,则命令行形式如下:framer input,其中,inputfile为原始数据文件,outputfile为输出结果。
•输出:对应input1和input2得结果分别为output1和output2。
试验要求:
•编写程序,根据给出的原始数据,组装一个IEEE
802.3格式的帧(题目)默认的输入文件为二进制原始数据(文件名分别为input1和input 2))。
•要求程序为命令行程序。比如,可执行文件名为framer.exe,则命令行形式如下:framer input,其中,inputfile为原始数据文件,outputfile为输出结果。
输出:对应input1和input2得结果分别为output1和output2
验设计相关知识:
帧:来源于串行线路上的通信。其中,发送者在发送数据的前后各添加特殊的字符,使它们成为一个帧。Ethernet从某种程度上可以被看作是机器之间的数据链路层连接。
按802.3标准的帧结构如下表所示(802.3标准的Ethernet帧结构由7部分组成)
其中,帧数据字段的最小长度为46B。如果帧的LLC数据少于46B,则应将数据字段填充至46B。填充字符是任意的,不计入长度字段值中。
在校验字段中,使用的是CRC校验。校验的范围包括目的地址字段、源地址字段、长度字段、LLC数据字段。
循环冗余编码(CRC)是一种重要的线性分组码、编码和解码方法,具有简单、检错和纠错能力强等特点,在通信领域广泛地用于实现差错控制。CRC校验码的检错能力很强,不仅能检查出离散错误,还能检查出突发错误。
利用CRC 进行检错的过程可简单描述如下:在发送端根据要传送的k 位二进制码序列,以一定的规则产生一个校验用的r 位监督码(CRC 码),附在原始信息的后边,构成一个新的二进制码序列(共k+r 位),然后发送出去。在接收端,根据信息码和CRC 码之间所遵循的规则进行检验,以确定传送中是否出错。这个规则在差错控制理论中称为“生成多项式”。
循环冗余校验码的特点:(1)CRC 校验码可检测出所有单个错误。(2)CRC 校验码可检测出所有奇数位错误。(3)CRC 校验码可检测出所有双位的错误(4)CRC 校验码可检测出所
有小于、等于校验位长度的突发错误。(5)CRC 校验码可以
](1/2)-[11
-k 的概率检测出长度为(K+1)位的突发错误
实验分析:
• 填充帧头部字段
要完成一次帧封装的过程,首先要完成的就是帧头部的装入,这一过程只要将签到吗、定界符、目的地址、源地址、长度字段的相应数值按顺序写入就可以了。其中,长度字段的值即为要发送的数据的实际长度。
• 填充数据字段
在填充数据字段的过程中要注意的主要问题是数据字段的长度。802.3标准中规定了帧数据字段的最小长度为46B ,最大长度为1500B 。如果数据不足46B ,则需要通过填充0来补足;若数据长度超过1500B ,则的大奖超过部分封装入下一个帧进行发送。
• CRC 校验
帧封装的最后一步就是对数据进行校验,并将校验结果记入帧校验字段。
程序流程图:
CRC计算流程图:
序源代码:
#include
#include
#include
void main(int argc,char*argv[])
{
//如果输入命令行不正确,则输出提示后退出。 if(argc!=3)
{
cout< exit(0); } //打开指定的输出文件,以二进制方式打开并可读可写,如文件存在,则清除其内容。 fstream [2],ios::out|ios::in|ios::binary|ios::trunc,0); for(int i=0;i<7;i++) ((char)0xaa); ((char)0xab);//写入B的前导码和B的帧前定界符。 char des_add[]={char(0x00),char(0x00),char(0xE4),char(0x86),char(0x3A),char(0xDC)}; (des_add,6);//写入B的目的地址。 char sor_add[]={char(0x00),char(0x00),char(0x80),char(0x1A),char(0xE6),char(0x65)}; (sor_add,6);//写入B的源地址。 //创建输入文件流并打开指定的输入文件,以二进制方式打开并可读。 ifstream in[1],ios::in|ios::binary,0); int length=0; in(0,ios::end);//将读指针移到文件末尾。 length=in();//计算指针偏移量,即为输入文件的长度。 unsigned char* data=new unsigned char[length];//创建字符指针并根据文件长度初始化。 in(0,ios::beg);//将读指针移到文件开始。 in(data,length);//将文件数据读入到字符指针data中。 (char(length>>8)); (char(length&0xff));//将文件长度值按照逆序写入到输出文件的长度字段中。 (data,length);//将data内容写入到输出文件中。 //如果输入文件长度不足B,则用补足B。 if(length<46) { for(int j=length;j<46;j++) (char(0x00)); } (char(0x00));//将数据字段后添加个 (8,ios::beg);//将读指针指向目的地址字段,从此处开始CRC计算 unsigned char ch;//ch用来保存读入的字符。 unsigned char crc=char(0x00);//余数初始值为。 while(1)//进行CRC计算 { (ch); if(ch==0xff)//判断是否到了文件结尾,如果是,则退出循环。 break; for(i=0;i<8;i++)//对入读入的字符的位分别处理。 { if(0x80==(crc&(0x80)))//当前余数最高位为,需要进行除法运算。 {