青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

大漠落日

while(!dead) study++;
posts - 46, comments - 126, trackbacks - 0, articles - 0
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

自己寫的一個ping程序

Posted on 2009-09-15 11:16 亂78糟 閱讀(6596) 評論(12)  編輯 收藏 引用 所屬分類: 網絡編程
以前學習ICMP協議時候寫的一個模仿windows自帶的ping程序,今天翻代碼看到了,貼上來供大家拍磚。


2010-12-15日更新補充:這段代碼獲取的是第一個活動網卡IP來發送ICMP報文,如果有多個網卡,例如插網線和wifi網卡同時開,那么就由插網線的網卡發送,如何讓wifi網卡發送呢?很簡單,加一個-p [bindLocalIP]參數就可以了,bindLocalIP就是指定綁定了某個IP網卡發送ICMP報文,而不是由gethostname來獲取,因為很簡單,所以代碼就懶得修改了,畢竟是多年前的代碼了。

本人已經在2K(32),XP(32),VISTA(64),WIN7(32)下測試過,一切正常。

    

/*******************************************************************
    
        程序:  myping
        功能:  模擬ping命令
        O/S :   WINDOWS 98或更高版本
        作者:  嚴政
        時間:  2007.8.14
        版本變更:
                V1.0.0   2007.8.14  初始版本,實現了基本的ping功能 
        說明:  
                這是開源代碼,你可以隨意拷貝使用。如果你有好的建議或
                意見,發E-mail至: [ yzljlss@126.com ]討論。
******************************************************************
*/
#include 
<stdio.h>
#include 
<WINSOCK2.H>

#pragma comment(lib,
"wsock32.lib")
#pragma comment(lib,
"Ws2_32.lib")

#define SIO_RECALL _WSAIOW(IOC_VENDOR,1)
#define MAX_HOSTNAME_LEN 256

sockaddr_in LocalAddr,SendAddr,destAddr;
SOCKET sock;

struct hostent FAR *pHostent, *pTmp;

int total = 4;//發送ping報文次數,默認為4次
int seconds = 0;//發送時間間隔
bool hostToIp = false;//-a 將目標的機器標識轉換為ip地址 
bool pingforever = false;//-t 若使用者不人為中斷會不斷的ping下去
bool isCount = false;//-c count 要求ping命令連續發送count個數據包
bool isSimple = false;//-q ping只在開始和結束時打印一些概要信息

char FAR name[MAX_HOSTNAME_LEN];
char destIP[16];//目標IP

typedef 
struct _ping
{
    UCHAR i_type;
//8位類型
    UCHAR i_code;//8位代碼
    USHORT i_chksum;//16位ICMP校驗和
    USHORT i_identify;//16位標志位
    USHORT i_seqnum;//16位序號
    ULONG    i_timestamp;//32位時間戳
    UCHAR i_data[32];//32BYTE選項數據
}PingHeader,*pPingHeader;

typedef 
struct     _ipHeader//IP頭部,總長度20字節
{
#if LITTLEENDIAN
    UCHAR IpHlen:
4,        //4位首部長度
          IpVer :4;        //4位IP版本號
#else
    UCHAR   IpVer :
4,        //4位IP版本號
            IpHlen:4;        //4位首部長度
#endif
    UCHAR  IpTos;
//8服務類型
    USHORT IpTlen;//總長度
    USHORT IpId;//標志    
    USHORT FlagsOff;//分片偏移
    UCHAR  IpTtl;//生存時間
    UCHAR  IpProto;//協議    
    USHORT ChkSum;//檢驗和
    struct in_addr   SourIp;//源IP地址    
    struct in_addr   DestIp; //目的IP地址
} IpHeader,*pIpHeader;

//求校驗和
USHORT checksum(USHORT *buffer, int size)
{
    unsigned 
long cksum = 0;
    
while(size > 1)
    {
        cksum 
+= *(buffer++);
        size 
-= sizeof(USHORT);
        
    }
    
if(size)
        cksum 
+= *(UCHAR*)buffer;
    cksum 
= (cksum >> 16+ (cksum & 0xffff);
    cksum 
+= (cksum >> 16);

    
return (USHORT)(~cksum);
}

//幫助信息
void help()
{
    printf(
"==========================[ myping V1.0.0 ]============================ ");
    printf(
" 用法: myping [-a] [-t] [-c count] [-i seconds] [-q] [-h] target_IP ");
    printf(
"參數: ");
    printf(
" -a     將目標的機器標識轉換為ip地址 ");
    printf(
"     建議: ping遠程主機時不要添加此參數,否則速度較慢 ");
    printf(
" -t     若使用者不人為中斷會不斷的ping下去 ");
    printf(
" -c count    要求ping命令連續發送count個數據包 ");
    printf(
" -i seconds  在兩次數據包發送之間間隔一定的秒數 ");
    printf(
" -q     myping只在開始和結束時打印一些概要信息 ");
    printf(
" -h     幫助信息 ");
    printf(
"例如: myping -a -i 1 -c 10 192.168.0.100 ");
    printf(
"=========================[ By 嚴政 07.8.14 ]=========================== ");
}

//發送,解析PING報文
int funPing()
{
    unsigned 
long i,
        totalrecv
=0;//收到包的數目
    unsigned long addr, timestamp,
        maxtime 
= 0,//最大延遲
        mintime = 0;//最小延遲
    PingHeader ping,*ping_hdr;
    
char recv_buff[65535];
    
char szDestIP[16];
    
int recvLen;

    pIpHeader ip_hdr;

    SendAddr.sin_family 
= AF_INET;
    SendAddr.sin_addr.s_addr 
= inet_addr(destIP);
    
    
if(hostToIp)
    {
        memset(name, 
0, MAX_HOSTNAME_LEN);
        
//獲取ping對象主機名
        addr = inet_addr(destIP);
        pHostent 
= gethostbyaddr((char *)&addr, sizeof(destIP) , AF_INET);
        
if(pHostent == NULL)
        {
            
//printf("fail to get host name: %d ",WSAGetLastError());
            fprintf(stdout, "Ping %s with 32 bytes of datas: ",destIP);
        }
        
else
            fprintf(stdout, 
"Ping %s[ %s ] with 32 bytes of datas: ", pHostent->h_name, destIP);
    }            
    
else
        fprintf(stdout, 
"Ping %s with 32 bytes of datas: ",destIP);
    
    
for(i=0;;i++)
    {
        
if(!pingforever)
        {
            
if(i >= (unsigned long)total)//達到發送次數
                break;
        }
        
        
//填充PING報文
        ping.i_type = 8;
        ping.i_code 
= 0;
        ping.i_seqnum 
= (USHORT)i;
        ping.i_identify 
= (unsigned short)GetCurrentProcessId(); 
        ping.i_timestamp 
= (unsigned long)::GetTickCount();
        
for(int j=0;j < 32; j++)
            ping.i_data[i] 
= (UCHAR)('a'+j);
        ping.i_chksum 
= 0;
        
//計算校驗和
        ping.i_chksum = checksum((unsigned short*)&ping,sizeof(ping));
        
//printf("checksum=%d ",ping.i_chksum);
        
        
if(sendto(sock, (char*)&ping, sizeof(ping),0, (struct sockaddr*)&SendAddr, sizeof(SendAddr)) == SOCKET_ERROR)
        {
            printf(
"Send ping packet error: %d ",WSAGetLastError());
            
return -1;
        }
        
        memset(recv_buff, 
01024);
        
int len = sizeof(destAddr);
        
if((recvLen = recvfrom(sock, recv_buff, sizeof(recv_buff), 0, (struct sockaddr*)&destAddr, &len)) == SOCKET_ERROR)
        {
            
int err = WSAGetLastError();
            
if(err != 10060)//超時錯誤不返回
            {
                printf(
"recv data error: %d ",err);
                
return -1;
            }
            
else if(!isSimple)
                fprintf(stdout, 
"請求超時. ");
        }
        
if(recvLen > 0)
        {
            
            
//處理接收的IP報文,解析PING應答報文
            ip_hdr = (pIpHeader)recv_buff;

            memcpy(szDestIP, inet_ntoa(ip_hdr
->SourIp), 16);
            
if(ip_hdr->IpProto == IPPROTO_ICMP && !strcmp(szDestIP, destIP))//處理來自PING對象且是ICMP的報文
            {
                ping_hdr 
= (pPingHeader)(recv_buff + sizeof(unsigned long)*ip_hdr->IpHlen);
                
            
//    fprintf(stdout,"ping_hdr.i_type=%02X ",ping_hdr->i_type);
            
//    fprintf(stdout,"ping_hdr.i_code=%02X ",ping_hdr->i_code);
            
//    fprintf(stdout,"ping_hdr.i_seqnum=%04X ",ping_hdr->i_seqnum);
            
//    fprintf(stdout,"ping_hdr.i_identify=%04X ",ping_hdr->i_identify);
            
//    fprintf(stdout,"ping_hdr.timestamp=%08X ",ping_hdr->i_timestamp);
                
//應答報文
                if(ping_hdr->i_type == 0)
                {
                    
//計算延遲時間
                    timestamp = (unsigned long)::GetTickCount();
                    timestamp 
-= ping_hdr->i_timestamp;
                    
if(i == 0)
                        mintime 
= timestamp;
                    maxtime 
= (timestamp > maxtime) ? timestamp : maxtime;//最大延遲時間
                    mintime = (timestamp < mintime) ? timestamp : mintime;//最小延遲時間

                    
if(timestamp == 0)
                        timestamp 
= 1;
                    
if(!isSimple)
                        fprintf(stdout, 
"Reply from %s: bytes=%d time<%dms TTL=%d ", destIP, 
                            
sizeof(ping_hdr->i_data), timestamp, ip_hdr->IpTtl );
                    
//收到包的數目
                    totalrecv++;
                }
                
if(ping_hdr->i_type == 3)
                {
                    fprintf(stdout, 
"目的不可達");
                    
switch(ping_hdr->i_code)
                    {
                    
case 0:
                        fprintf(stdout, 
"(網絡不可達) ");
                        
break;
                    
case 1:
                        fprintf(stdout, 
"(主機不可達) ");
                        
break;
                    
case 2:
                        fprintf(stdout, 
"(協議不可達) ");
                        
break;
                    
case 3:
                        fprintf(stdout, 
"(端口不可達) ");
                        
break;
                    
default:
                        
break;
                    }
                }
                
if(ping_hdr->i_type == 5)
                {
                    
if(ping_hdr->i_code == 0)
                        fprintf(stdout, 
"對網絡重定向. ");
                    
if(ping_hdr->i_code == 1)
                        fprintf(stdout, 
"對主機重定向. ");
                }
                
            }
        }
        Sleep(seconds);
    }
//end for
    
    
//計算ping統計信息
    fprintf(stdout, " Ping %s 的統計信息: ",destIP);
    fprintf(stdout, 
" 包: 發送 = %d, 收到 = %d, 丟失 = %d (丟失率: %.0f%%) ",
        i, totalrecv, (i
-totalrecv), ((float)(i-totalrecv))/i*100 );
    
if(totalrecv != 0)//沒收到包打印路由信息無意義
    {
        fprintf(stdout, 
"近似路由時間(毫秒): ");
        fprintf(stdout, 
" 最大 = %dms, 最小 = %dms, 平均 = %dms ", maxtime, mintime, (maxtime+mintime)/2);
    }
    
return 0;
}

int main(int argc, char* argv[])
{
    DWORD lpvBuffer 
= 1;
    DWORD lpcbBytesReturned 
= 0;
    
    
int nNetTimeout=3000;//超時3秒
    
    
//初始化套結字
    WSADATA WSAData;
    
if(WSAStartup(MAKEWORD(2,2), &WSAData) != 0)
    {
        printf(
"fail to init socket: %d",WSAGetLastError());
        
return -1;
    }
    
//創建套結字
    sock = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);
    setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO, (
char *)&nNetTimeout,sizeof(int));

    
if(sock == SOCKET_ERROR)
    {
        printf(
"fail to create socket: %d",WSAGetLastError());
        
return -1;
    }
    
//獲取本機IP
    if(gethostname(name, MAX_HOSTNAME_LEN))
    {
        printf(
"get host name error: %d",WSAGetLastError());
        
return -1;
    }
    pHostent 
= (struct hostent *)malloc(sizeof(struct hostent));
    pTmp 
= pHostent;
    pHostent 
= gethostbyname(name);
 
    LocalAddr.sin_family 
= AF_INET;
    LocalAddr.sin_port 
= htons(0);
    memcpy(
&LocalAddr.sin_addr.S_un.S_addr, pHostent->h_addr_list[0], pHostent->h_length);
    
    
//bind socket
    if(bind(sock, (struct sockaddr *)&LocalAddr, sizeof(LocalAddr)) == SOCKET_ERROR)
    {
        printf(
"bind error: %d",WSAGetLastError());
        
return -1;
    }
/*    
    //置網卡為混雜模式
    //在這個ping程序中,不能將網卡置于混雜模式,否則接收的報文包括自己發送的報文 (^_^)
    if(WSAIoctl(sock, SIO_RECALL, &lpvBuffer ,sizeof(lpvBuffer), NULL, 0,
        &lpcbBytesReturned, NULL, NULL) == SOCKET_ERROR)
    {
        printf("WSAIoctl() error: %d",WSAGetLastError());
        return -1;
    }
*/    
    
//命令行解析
    if(argc < 2)
    {
        printf(
"必須輸入ping參數! ");
        help();
        
return -1;
    }
    
else
    {
        
for(int i=1; i < argc; i++)
        {
            
if(argv[i][0== '-')
            {
                
switch( (tolower(argv[i][1])) )
                {
                
case 'a':
                    hostToIp 
= true;
                    
break;
                
case 't':
                    pingforever 
= true;
                    
break;
                
case 'h':
                    help();
                    
return -1;
                    
break;
                
case 'q'://ping只在開始和結束時打印一些概要信息
                    isSimple = true;
                    
break;
                
case 'c':
                    {
                        
if*(argv[i]+3> '9' || *(argv[i]+3< '0' )
                        {
                            printf(
"ping次數錯誤參數! ");
                            help();
                            
return -1;
                        }
                        
//發送報文次數
                        total = atoi(argv[++i]);
                        
break;
                    }
                
case 'i'://設置發送報文時間差
                    {
                        
if*(argv[i]+3> '9' || *(argv[i]+3< '0' )
                        {
                            printf(
"時間錯誤參數! ");
                            help();
                            
return -1;
                        }
                        
//發送報文次數
                        seconds = atoi(argv[++i]) * 1000;
                        
break;
                    }
                
default:
                    
break;
                }
//end switch
            }//end if
        }//end for
        
        
if( (argv[argc-1][0]) > '9' || (argv[argc-1][0]) < '0' )
        {
            printf(
"目的IP錯誤!請確認最后一個參數是目的IP. ");
            help();
            
return -1;
        }
        
        memcpy(destIP, argv[argc
-1], strlen(argv[argc-1]));
    }
    
//發送ping報文
    funPing();

    free(pTmp);
    closesocket(sock);
    WSACleanup();

    
return 0;
}


Feedback

# re: 自己寫的一個ping程序[未登錄]  回復  更多評論   

2009-09-15 17:56 by foxriver
為了安全因素,SOCK_RAW在現在的操作系統上不支持了吧,只能用IMCP開頭的API。

個人網絡項目里,ping值一直是用socket連接后,發個小包計算返回時間,實際測試效果也不錯。

# re: 自己寫的一個ping程序  回復  更多評論   

2009-09-16 15:25 by 羅萊家紡
搞得不錯啊~

# re: 自己寫的一個ping程序  回復  更多評論   

2009-09-18 11:43 by 亂78糟
@foxriver
VISTA之后就禁止了,XP下沒問題的。

非常感謝提醒返回時間的計算方法

# re: 自己寫的一個ping程序  回復  更多評論   

2009-09-24 09:54 by 過客
for(int j=0;j < 32; j++)
ping.i_data[i] = (UCHAR)('a'+j);

循環的ping.i_data[i] or ping.i_data[j]?

# re: 自己寫的一個ping程序  回復  更多評論   

2009-09-24 11:00 by 亂78糟
@過客
因為ping報文的數據段可以使任意數據,所以填什么無所謂,不過的確是我寫錯了,應該是j,不是i

# re: 自己寫的一個ping程序  回復  更多評論   

2011-05-31 16:43 by 游客
為什么我在win7下運行程序直接到末尾了,沒報錯什么的,直接就結束,沒有提示輸入參數

# re: 自己寫的一個ping程序  回復  更多評論   

2012-05-02 09:54 by 吳同學
十分感謝分享,正在學習

# re: 自己寫的一個ping程序  回復  更多評論   

2014-03-04 10:35 by 黑魚
象天書,完全不懂

# re: 自己寫的一個ping程序  回復  更多評論   

2015-04-23 16:59 by 8
@游客
我的也是

# re: 自己寫的一個ping程序  回復  更多評論   

2015-04-23 17:02 by 8
程序直接進入
if(argc < 2)
{
printf("必須輸入ping參數! ");
help();
return -1;
}
循環后結束,提示輸入參數,但是應該怎樣輸入參數?謝謝!

# re: 自己寫的一個ping程序  回復  更多評論   

2015-06-26 16:46 by smagle
@8
help()里有提示的,例如:myping -a -i 1 -c 10 192.168.0.100

# re: 自己寫的一個ping程序  回復  更多評論   

2016-03-23 17:24 by gu*

@游客
我的也是這種情況怎么回事啊到底

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品每日更新| 欧美激情aⅴ一区二区三区| 国产精品日本| 午夜精品美女久久久久av福利| 香蕉久久一区二区不卡无毒影院 | 亚洲摸下面视频| 欧美一区二区网站| 激情成人综合网| 欧美二区在线看| 亚洲性夜色噜噜噜7777| 久久国产精品网站| 亚洲激精日韩激精欧美精品| 欧美日韩国产在线播放网站| 亚洲免费在线观看视频| 久久综合精品一区| 一区二区动漫| 国产综合精品一区| 欧美激情精品久久久久久大尺度 | 影音先锋另类| 欧美激情精品久久久六区热门| 99re66热这里只有精品4| 久久xxxx精品视频| 亚洲精品婷婷| 国内精品久久久| 欧美精品不卡| 久久激情视频免费观看| 91久久极品少妇xxxxⅹ软件| 先锋影音久久久| 亚洲精品色婷婷福利天堂| 国产精品色在线| 欧美激情一二区| 久久国产日韩欧美| 一区二区三区黄色| 欧美国产日韩一区二区| 欧美一区日本一区韩国一区| 亚洲福利视频二区| 国产日韩成人精品| 欧美性片在线观看| 农村妇女精品| 欧美在线观看一区| 中文一区二区在线观看| 欧美激情精品久久久六区热门| 欧美一级专区| 一本色道久久综合亚洲精品高清 | 另类综合日韩欧美亚洲| 亚洲视频一二| 亚洲美女视频在线观看| 一区二区三区在线高清| 国产精品手机视频| 欧美日韩国产一级| 欧美不卡一卡二卡免费版| 久久福利资源站| 欧美亚洲免费高清在线观看| 99国产精品视频免费观看| 亚洲电影av| 欧美www在线| 免费看成人av| 久久亚洲影院| 久热精品视频在线| 久久色中文字幕| 久久久在线视频| 久久精品国亚洲| 欧美主播一区二区三区美女 久久精品人 | 欧美高清在线播放| 久久一区二区三区国产精品| 久久精视频免费在线久久完整在线看| 亚洲一卡久久| 亚洲一二三四久久| 亚洲免费在线视频一区 二区| 99视频精品免费观看| 亚洲精品国产精品乱码不99| 亚洲激情在线观看| 亚洲激情在线观看视频免费| 亚洲福利视频专区| 亚洲黄色成人| 亚洲青色在线| 在线视频你懂得一区二区三区| 亚洲伦理精品| 国产精品99久久久久久宅男| 亚洲图片你懂的| 亚洲欧美国产视频| 久久国产精品99精品国产| 久久久久九九九| 老司机午夜免费精品视频| 免费在线观看成人av| 欧美精品在线视频观看| 欧美视频官网| 国产精品毛片一区二区三区| 国产九区一区在线| 狠狠色丁香婷婷综合影院| 亚洲高清在线| 亚洲特色特黄| 欧美一区二区视频免费观看| 久久精品国产第一区二区三区最新章节| 久久精品国产一区二区三区| 麻豆av福利av久久av| 亚洲国产成人精品女人久久久| 日韩视频在线免费| 亚洲欧美日韩一区二区在线| 亚洲黄色一区| 亚洲激情偷拍| 一区二区激情| 欧美一区精品| 欧美激情在线狂野欧美精品| 欧美特黄一区| 影音先锋亚洲电影| 中日韩午夜理伦电影免费| 欧美在线视频全部完| 亚洲第一中文字幕在线观看| 这里只有精品电影| 久久久久久久一区| 欧美日韩一区二区在线| 国产亚洲精品久久久久久| 最近中文字幕日韩精品| 西瓜成人精品人成网站| 亚洲大胆人体视频| 亚洲影音一区| 久久天天躁夜夜躁狠狠躁2022 | 亚洲电影免费观看高清完整版在线观看 | 99热这里只有精品8| 午夜欧美精品| 亚洲国产网站| 亚洲一区激情| 乱人伦精品视频在线观看| 欧美午夜精品久久久久久孕妇 | 在线观看欧美黄色| 亚洲综合久久久久| 久久综合国产精品| 亚洲美女视频在线观看| 久久精品在线观看| 国产精品美女在线观看| 亚洲国产精品久久久久婷婷老年 | 性欧美暴力猛交69hd| 亚洲高清av在线| 午夜亚洲福利| 欧美日韩日本网| 最新中文字幕亚洲| 久久亚洲精选| 欧美一级夜夜爽| 午夜激情综合网| 午夜精品亚洲| 国产精品国产三级国产普通话蜜臀| 在线成人激情| 久久精品国产久精国产思思| 一本色道久久88综合日韩精品 | 国产精品一区二区久久久久| 亚洲精品一区久久久久久| 久久精品国产免费观看| 一区二区三区久久精品| 欧美国产欧美亚洲国产日韩mv天天看完整 | 亚洲一二三级电影| 亚洲黄色性网站| 免费成人av在线| 亚洲第一视频网站| 久久深夜福利免费观看| 午夜免费在线观看精品视频| 国产精品久久久久久模特| 在线亚洲伦理| 9国产精品视频| 欧美三级午夜理伦三级中视频| 99国产精品99久久久久久粉嫩| 亚洲第一级黄色片| 欧美成人首页| 日韩亚洲欧美成人一区| 亚洲精品久久久久中文字幕欢迎你| 欧美aaaaaaaa牛牛影院| 91久久精品一区| 亚洲激情影视| 欧美日韩在线观看一区二区三区| 一本色道久久精品| 日韩视频在线一区| 欧美性片在线观看| 欧美一区二视频| 性色av一区二区三区在线观看 | 亚洲在线视频免费观看| 国产精品久久一卡二卡| 欧美一级网站| 久久精品人人做人人爽电影蜜月| 黄色影院成人| 欧美国内亚洲| 欧美日韩国产色综合一二三四 | 美女图片一区二区| 久久综合99re88久久爱| 亚洲激情视频网站| 亚洲免费高清| 国产日韩欧美高清| 欧美承认网站| 欧美日韩一区二区在线视频| 欧美一区二区三区另类| 久久成人免费日本黄色| 亚洲激情国产| 亚洲一区二区三区免费在线观看 | 中文av一区特黄| 午夜精品国产更新| 亚洲丰满少妇videoshd| 亚洲毛片在线看| 国产日本欧美一区二区三区在线 | 激情视频一区二区三区| 亚洲国产精品福利| 国产精品乱看|