/*导入库文件*/
#pragma comment( lib, "ws2_32.lib" )
/*加载头文件*/
#include <winsock2.h>
#include <ws2tcpip.h>
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
/* 【自学去】网站收集 http://www.zixue7.com */
/*定义常量*/
/*表示要记录路由*/
#define IP_RECORD_ROUTE 0x7
/*默认数据报大小*/
#define DEF_PACKET_SIZE 32
/*最大的ICMP数据报大小*/
#define MAX_PACKET 1024
/*最大IP头长度*/
#define MAX_IP_HDR_SIZE 60
/*ICMP报文类型,回显请求*/
#define ICMP_ECHO 8
/*ICMP报文类型,回显应答*/
#define ICMP_ECHOREPLY 0
/*最小的ICMP数据报大小*/
#define ICMP_MIN 8
/*自定义函数原型*/
void InitPing();
void UserHelp();
void GetArgments(int argc, char** argv);
USHORT CheckSum(USHORT *buffer, int size);
void FillICMPData(char *icmp_data, int datasize);
void FreeRes();
void DecodeIPOptions(char *buf, int bytes);
void DecodeICMPHeader(char *buf, int bytes, SOCKADDR_IN* from);
void PingTest(int timeout);
/*IP报头字段数据结构*/
typedef struct _iphdr
{
unsigned int h_len:4; /*IP报头长度*/
unsigned int version:4; /*IP的版本号*/
unsigned char tos; /*服务的类型*/
unsigned short total_len; /*数据报总长度*/
unsigned short ident; /*惟一的标识符*/
unsigned short frag_flags; /*分段标志*/
unsigned char ttl; /*生存期*/
unsigned char proto; /*协议类型(TCP、UDP等)*/
unsigned short checksum; /*校验和*/
unsigned int sourceIP; /*源IP地址*/
unsigned int destIP; /*目的IP地址*/
} IpHeader;
/*ICMP报头字段数据结构*/
typedef struct _icmphdr
{
BYTE i_type; /*ICMP报文类型*/
BYTE i_code; /*该类型中的代码号*/
USHORT i_cksum; /*校验和*/
USHORT i_id; /*惟一的标识符*/
USHORT i_seq; /*序列号*/
ULONG timestamp; /*时间戳*/
} IcmpHeader;
/*IP选项头字段数据结构*/
typedef struct _ipoptionhdr
{
unsigned char code; /*选项类型*/
unsigned char len; /*选项头长度*/
unsigned char ptr; /*地址偏移长度*/
unsigned long addr[9]; /*记录的IP地址列表*/
} IpOptionHeader;
/*定义全局变量*/
SOCKET m_socket;
IpOptionHeader IpOption;
SOCKADDR_IN DestAddr;
SOCKADDR_IN SourceAddr;
char *icmp_data;
char *recvbuf;
USHORT seq_no ;
char *lpdest;
int datasize;
BOOL RecordFlag;
double PacketNum;
BOOL SucessFlag;
/*初始化变量函数*/
void InitPing()
{
WSADATA wsaData;
icmp_data = NULL;
seq_no = 0;
recvbuf = NULL;
RecordFlag = FALSE;
lpdest = NULL;
datasize = DEF_PACKET_SIZE;
PacketNum = 5;
SucessFlag = FALSE;
/*Winsock初始化*/
if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
{
/*如果初始化不成功则报错,GetLastError()返回发生的错误信息*/
printf("WSAStartup() failed: %d\n", GetLastError());
return ;
}
m_socket = INVALID_SOCKET;
}
/*显示信息函数*/
void UserHelp()
{
printf("UserHelp: ping -r <host> [data size]\n");
printf(" -r record route\n");
printf(" -n record amount\n");
printf(" host remote machine to ping\n");
printf(" datasize can be up to 1KB\n");
ExitProcess(-1);
}
/*获取ping选项函数*/
void GetArgments(int argc,char** argv)
{
int i;
int j;
int exp;
int len;
int m;
/*如果没有指定目的地地址和任何选项*/
if(argc == 1)
{
printf("\nPlease specify the destination IP address and the ping option as follow!\n");
UserHelp();
}
for(i = 1; i < argc; i++)
{
len = strlen(argv[i]);
if (argv[i][0] == '-')
{
/*选项指示要获取记录的条数*/
if(isdigit(argv[i][1]))
{
PacketNum = 0;
for(j=len-1,exp=0;j>=1;j--,exp++)
/*根据argv[i][j]中的ASCII值计算要获取的记录条数(十进制数)*/
PacketNum += ((double)(argv[i][j]-48))*pow(10,exp);
}
else
{
switch (tolower(argv[i][1]))
{
/*选项指示要获取路由信息*/
case 'r':
RecordFlag = TRUE;
break;
/*没有按要求提供选项*/
default:
UserHelp();
break;
}
}
}
/*参数是数据报大小或者IP地址*/
else if (isdigit(argv[i][0]))
{
for(m=1;m<len;m++)
{
if(!(isdigit(argv[i][m])))
{
/*是IP地址*/
lpdest = argv[i];
break;
}
/*是数据报大小*/
else if(m==len-1)
datasize = atoi(argv[i]);
}
}
/*参数是主机名*/
else
lpdest = argv[i];
}
}
/*求校验和函数*/
USHORT CheckSum(USHORT *buffer, int size)
{
unsigned long cksum=0;
while (size > 1)
{
cksum += *buffer++;
size -= sizeof(USHORT);
}
if (size)
{
cksum += *(UCHAR*)buffer;
}
/*对每个16bit进行二进制反码求和*/
cksum = (cksum >> 16) + (cksum & 0xffff);
cksum += (cksum >>16);
return (USHORT)(~cksum);
}
/*填充ICMP数据报字段函数*/
void FillICMPData(char *icmp_data, int datasize)
{
IcmpHeader *icmp_hdr = NULL;
char *datapart = NULL;
icmp_hdr = (IcmpHeader*)icmp_data;
/*ICMP报文类型设置为回显请求*/
icmp_hdr->i_type = ICMP_ECHO;
icmp_hdr->i_code = 0;
/*获取当前进程IP作为标识符*/
icmp_hdr->i_id = (USHORT)GetCurrentProcessId();
icmp_hdr->i_cksum = 0;
icmp_hdr->i_seq = 0;
datapart = icmp_data + sizeof(IcmpHeader);
/*以数字0填充剩余空间*/
memset(datapart,'0',datasize-sizeof(IcmpHeader));
}
/*释放资源函数*/
void FreeRes()
{
/*关闭创建的套接字*/
if (m_socket != INVALID_SOCKET)
closesocket(m_socket);
/*释放分配的内存*/
HeapFree(GetProcessHeap(), 0, recvbuf);
HeapFree(GetProcessHeap(), 0, icmp_data);
/*注销WSAStartup()调用*/
WSACleanup();
return ;
}
/*解读IP选项头函数*/
void DecodeIPOptions(char *buf, int bytes)
{
IpOptionHeader *ipopt = NULL;
IN_ADDR inaddr;
int i;
HOSTENT *host = NULL;
/*获取路由信息的地址入口*/
ipopt = (IpOptionHeader *)(buf + 20);
printf("RR: ");
for(i = 0; i < (ipopt->ptr / 4) - 1; i++)
{
inaddr.S_un.S_addr = ipopt->addr[i];
if (i != 0)
printf(" ");
/*根据IP地址获取主机名*/
host = gethostbyaddr((char *)&inaddr.S_un.S_addr,sizeof(inaddr.S_un.S_addr), AF_INET);
/*如果获取到了主机名,则输出主机名*/
if (host)
printf("(%-15s) %s\n", inet_ntoa(inaddr), host->h_name);
/*否则输出IP地址*/
else
printf("(%-15s)\n", inet_ntoa(inaddr));
}
return;
}
/*解读ICMP报头函数*/
void DecodeICMPHeader(char *buf, int bytes, SOCKADDR_IN *from)
{
IpHeader *iphdr = NULL;
IcmpHeader *icmphdr = NULL;
unsigned short iphdrlen;
DWORD tick;
static int icmpcount = 0;
iphdr = (IpHeader *)buf;
/*计算IP报头的长度*/
iphdrlen = iphdr->h_len * 4;
tick = GetTickCount();
/*如果IP报头的长度为最大长度(基本长度是20字节),则认为有IP选项,需要解读IP选项*/
if ((iphdrlen == MAX_IP_HDR_SIZE) && (!icmpcount))
/*解读IP选项,即路由信息*/
DecodeIPOptions(buf, bytes);
/*如果读取的数据太小*/
if (bytes < iphdrlen + ICMP_MIN)
{
printf("Too few bytes from %s\n",
inet_ntoa(from->sin_addr));
}
icmphdr = (IcmpHeader*)(buf + iphdrlen);
/*如果收到的不是回显应答报文则报错*/
if (icmphdr->i_type != ICMP_ECHOREPLY)
{
printf("nonecho type %d recvd\n", icmphdr->i_type);
return;
}
/*核实收到的ID号和发送的是否一致*/
if (icmphdr->i_id != (USHORT)GetCurrentProcessId())
{
printf("someone else's packet!\n");
return ;
}
SucessFlag = TRUE;
/*输出记录信息*/
printf("%d bytes from %s:", bytes, inet_ntoa(from->sin_addr));
printf(" icmp_seq = %d. ", icmphdr->i_seq);
pri
CrMylive.
- 粉丝: 1w+
- 资源: 4万+
最新资源
- 有向图下多智能体自适应二分时变编队控制;多智能体编队;自适应二分时变;有向图(有参考文献)
- 基于mpc的燃料电池混合动力汽车能量管理策略 1.研究对象为燃料电池-动力电池混合动力汽车 2.假设预测域内车速已知,在模型预测控制框架下构建最优控制问题 3.分别利用动态规划和pmp求解预测域内的能
- 双功率通路双向DC-AC变器 直流变器采用非隔离双向Buck Boost变器拓扑,DC-AC变器采用T型拓扑 该电路拓扑能够实现蓄电池和单相交流电网之间部分功率的单级变,且具备双向功率传输能力,适
- 三电平npc有源电力滤波器无差拿控制matlab 仿真
- 电机马达本杰明磁链模型,可以闭环带载启动,支持代码生成,已经实际电机验证
- Fluent焊接熔池模拟,圆弧焊接,自定义热源路径,圆弧焊接 ICEM建模与网格划分
- 欧姆龙NJ501 PLC新能源锂电行业激光焊接程序案例 程序26轴EtherCAT总线控制,伺服使用的是汇川伺服 设备工19个工位,程序分工位分模块编程,逻辑清晰,注释齐全,可读性较好 自编进20
- 三电平无刷直流电机BLDC矢量控制仿真模型,给定转速1000r min,运行良好; 五电平,两电平均可做,可调参数; matlab simulink模型
- 含风电-光伏-光热电站电力系统N-k安全优化调度模型该程序参考《光热电站促进风电消纳的电力系统优化调度》光热电站模型,主要做的是考虑N-k安全约束的含义风电-光伏-光热电站的电力系统优化调度模型,从而
- BLDC(无刷直流电机)矢量控制仿真模型,在转速1500r min运行良好,可升级为滑模控制; 也有采用霍尔的六步相双闭环控制;
- pemfc燃料电池电堆fluent仿真 带冷却流道的燃料电池电堆,热管理仿真 燃料电池电堆三维模型创建和fluent流场仿真教程 包含模型,网格,算例和结果文件
- 三菱和MCGS自动洗衣机控制系统组态模拟仿真控制系统组态王PLC程序
- 多智能体自适应时变编队跟踪控制;编队跟踪;多智能体;观测器
- MATLAB环境下一种改进盲反卷积算法 算法运行环境为MATLAB R2018A,执行一种改进盲反卷积算法,可用于旋转机械故障诊断 压缩包=代码+数据+参考 subplotnum-1 = 3; s
- 整车七自由度主动悬架模型 基于simulik搭建的整车七自由度主动悬架模型,采用模糊PID控制策略,以悬架主动力输入为四轮随机路面,输出为平顺性评价指标垂向加速度等,悬架主动力为控制量,车身垂向速度为
- 全局规划算法 rrt+dubins规划 hybrid astar规划 c++实现栅格地图,状态空间,状态校验,RRT搜索算法 混合a星规划算法
资源上传下载、课程学习等过程中有任何疑问或建议,欢迎提出宝贵意见哦~我们会及时处理!
点击此处反馈