当前位置:文档之家› 网络通信协议分析课程设计源代码和实验报告+帧封装、IP数据包解析和发送TCP数据包

网络通信协议分析课程设计源代码和实验报告+帧封装、IP数据包解析和发送TCP数据包

网络协议分析课程设计之协议编程

实验一帧封装

实验目的:

•编写程序,根据给出的原始数据,组装一个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)))//当前余数最高位为,需要进行除法运算。

{

相关主题