计算机网络课程设计报告--发送TCP数据包设计.docx
计算机网络课程设计报告发送TCP数据包一、问题描述经过我们的慎重思考和搜集各个问题的资料及与其他同学的交流,我们所选的问题是发送TCP数据包。众所周知,TCP(传输控制协议)是一种面向连接的、可靠的传输层协议。TCP协议在网络层IP协议的基础上,向应用层用户进程提供可靠的、全双工的数据传输流。本课程设计的目的就是设计一个发送TCP数据包的程序,并根据本设计说明TCP数据包的结构以及TCP协议与IP协议的关系,使大家对TCP协议的工作原理有更深入的认识。二、概要设计(抽象数据类型定义)typedefstructip_hdr定义IP首部UCHARh_ver1en;UCHARtos;USHORTtota1Jen;USHORTident;4位首部长度,4位IP版本号/8位服务类型ToS/16位总长度(字节)16位标识USHORTfrag_and_f1ags;UCHARtt1;/3位标志位8位生存时间TT1UCHARproto;USHORTchecksum;/8位协议(TCP,UDP或其它)16位IP首部校验和U1ONGsourceIP;U1ONGdestIP;/32位源IP地址32位目的IP地址IP_HEADER;typedefstructtsd_hdr定义TCP伪首部U1ONGsaddr;U1ONGdaddr;UCHARmbz;UCHARptc1;USHORTtcp1;源地址目的地址没用协议类型/TCP长度PSD.HEADER;typedefstructtcp_hdr(USHORTth.sport;USHORTth-dport;U1ONGth_seq;U1ONGth_ack;UCHARthjenres;UCHARth.f1ag;USHORTth_win;USHORTth.sum;USHORTth.urp;TCP.HEADER;定义TCP首部/16位源埠16位目的端口/32位序列号/32位确认号4位首部长度/6位保留字/6位标志位/16位窗口大小16位校验和/16位紧急数据偏移量三、详细设计(主要算法和函数间的调用关系)3.1使用原始套接字SOCKETsock;SOCk=SoCket(AFNET,SOCK_RAWJPPROTOP);或者用下面的定义写:sock=WSASoccket(AF_INET,SOCK_RAW,IPPROTO_IP,NU11,0,WSA_F1AG_OVER1APPD);这里,我们设置了SoCK_RAW标志,表示我们声明的是一个原始套接字类型。为使用发送接收超时设置,必须将标志位置置为WSA_F1AG_OVER1APPED“在本课程设计中,发送TCP包时陷藏了自己的IP地址,因此我们要自己填充IP头,设置IP头操作选项。3.2计算校验和的子函数。在填充数据包的过程中,需要调用计算校验和的函数CheekSUm两次,分别用于校验IP头和TCP头部(加上伪头部),其实现代码如下:USHORTchecksum(USHORT*buffer,intsize)unsigned1ongcksum=0;/累加和初始值Whi1e(Sizo1)多少个USHORT型数据(cksum+=*buffer+;算计累加和size.=Sizeof(USHORT);个数-1)if(size)如果有非对齐的下余数据(cksum+=*(UCHAR*)buffer;十余卜的数值cksum=(cksum»16)+(cksum&OXffff);右移16位,得到前16位,再力口上cksum&Oxffff,Oxffff就是全是1,就是cksum等于cksum加上他的前16位。cksum+=(cksum»16);然后cksum等于cksum力口上cksum的前16位。return(USHORT)(-cksum);/返回它的反(USHORT型)3.3程序流程图源代码#inc1ude<stdio.h>#inc1ude<winsock2.h>#inc1ude<ws2tcpip.h>套接字头文件设置或获取,链接套接字#inc1ude<time.h>#inc1ude<windows.h>重要的头文件,包含基本数据类型定义,图形设备接口函数等#inc1ude<string.h>#inc1ude<std1ib.h>#inc1ude<iostream.h>#pragmaCOmment(IibJWS2_32.1ib")#defineIPVER4/ZIP协议预定#defineMAX_BUFF_1EN65500发送缓冲区最大值typedefstructip_hdrf定义IP首部IUCHARh_ver1en;UCHARtos;USHORTtota1Jen;USHORTident;4位首部长度,4位IP版本号/8位服务类型TOS16位总长度(字节)16位标识USHORTfrag_and_f1ags;/3位标志位UCHARtt1;UCHARproto;USHORTchecksum;/8位生存时间TT1/8位协议(TCP,UDP或其它)16位IP首部校验和U1ONGsourceIP;U1ONGdestIP;IP_HEADER;/32位源IP地址/32位目的IP地址typedefstructtsd_hdr定义TCP伪首部(U1ONGsaddr;源地址U1ONGdaddr;目的地址UCHARmbz;没用UCHARptc1;协议类型USHORTtcp1:/TCP长度PSD.HEADER;typedefstructtcp_hdr定义TCP首部USHORTth.sport;/16位源端口USHORTth_dport;U1ONGth.seq;16位目的端口Z32位序列号U1ONGth_ack;UCHARth_1enres;/32位确认号4位首部长度/6位保留字UCHARth_f1ag;/6位标志位USHORTth.win;/16位窗口大小USHORTth-sum;USHORTth.urp;16位校验和16位紧急数据偏移量TCP.HEADER;/CheckSum:计算校验和的子函数USHORTchecksum(USHORT*buffer,intsize)(unsigned1ongcksum=O;累加和初始值Whi1e(Sizo1)多少个USHORT型数据(cksum+=*buffer+;计算累加和size-=Sizeof(USHORT);个数-1)if(size)余下的数据(cksum+=*(UCHAR*)buffer;+余下的数值)cksum=(cksum»16)+(cksum&Oxffff);右移16位,得至!前16位,再力口上cksum&Oxffff,Oxffff就是全是1,就是cksum等于cksum加上他的前16位。cksum+=(cksum»16);然后CkSUm等于CkSUnI加上CkSUnI的前16位。return(USHORT)(-cksum);返回它的反(USHORT型)intmain(intargc,char*argv)(WSADAWSAData;SOCKETsock;IP_HEADERipHeader;TCP-HEADERtcpHeader;PSD_HEADERpsdHeader;charSendto-Buff1MAX-BUFF-1EN;发送缓冲区unsignedshortcheck_BuffMAX_BUFF_1EN;检验缓冲区constchartcp_send_data="Thisismyhomeworkofnetwort,Iamhappy!"BOO1f1ag;intrect,nTimeOver;if(argc!=5)(printfC,Useage:SendTcpsoruce_ipsource_portdest_ipdest_portn);returnfa1se;)if(WSAStartup(MAKEWORD(2,2),&WSAData)!=O)(printf(HWSAStartupError!n);returnfa1se;)if(sock=WSASocket(AFNET,SOCK_RAWJPPROTO_RAW,NU11,0,WSA_F1AG_OVER1APPED)=INVA1ID_SOCKET)printf(',SocketSetupError!n'');returnfa1se;)f1ag=true;if(setsockopt(sock,IPPROTO_IP,IP_HDRINC1,(char*)&f1ag,sizeof(f1ag)=S0CKET_ERR0R)(printf(,setsockopt1P_HDRINC1error!n");returnfa1se;)nTimeOver=1000;if(setsockopt(sock,SO1.SOCKET,SO_SNDTIMEO,(char*)&nTimeOver,sizeof(nTimeOver)=SOCKET_ERROR)(printf(',setsockoptSO_SNDTIMEOerror!n,');returnfa1se;)填充IP首部ipHeader.h_ver1en=(IPVER«4sizeof(ipHeader)sizeof(unsigned1ong);ipHeader.tos=(UCHAR)0;ipHeader.tota1Jen=htons(unsignedshort)sizeof(ipHeader)+sizeof(tcpHeader)+sizeof(tcp_send_data);IpHeaderJdent=O;16位标识ipHeader.frag_and_f1ags=O;/3位标志位ipHeader.tt1=128;/8位生存时间ipHeader.proto=IPPROTOjJDP;/协议类型ipHeader.checksum=O;/检验和暂时为0ipHeader.sourceIP=inet-addr(argv1);/32位源IP地址ipHeader.dest1P=inet-addr(argv3);/32位目的IP地址计算IP头部检验和memset(check_Buff,O,MAX_BUFF_1EN);memcpy(check_Buff,&ipHeader,sizeof(IP_HEADER);ipHeader.checksum=checksum(check_Buff,sizeof(IP_HEADER);/构造TCP伪首部psdHeader.saddr=ipHeader.sourceIP;psdHeader.daddr=ipHeader.destIP;psdHeader,mbz=0;psdHeader.ptc1=ipHeader.proto;psdHeader.tcp1=htons(sizeof(TCP_HEADER)+sizeof(tcp_