新客网WWW.XKER.COM:致力做中国最专业的网络学院!
学院: 操作系统 - 网络应用 - 服务器 - 网络安全 - 工具软件 - 办公软件 - Web开发 - 数据库 - 网页设计 - 图形图像 - 媒体动画 - 硬件学堂 - 存储频道 - QQ专区
您的位置:首页 > 网络学院 > 操作系统 > Linux教程 > 正文:Linux实用技巧 用ICMP实现简单的Ping功能

Linux实用技巧 用ICMP实现简单的Ping功能

新客网 XKER.COM 2007-08-01 来源: kit 收藏本文

如果目的主机在工 输出在工状态 如果5妙内无相应 用SIGALRM信号中断进程

#include "unp.h"

void send_echo_req(int sockfd, struct sockaddr_in *dstaddr);
uint16_t in_cksum(uint16_t *addr, int len);
void recv_echo_reply(int sockfd);

int main(int argc, char **argv)
{
        int sockfd;
        struct sockaddr_in dstaddr;

        if ((sockfd = socket(PF_INET, SOCK_RAW, IPPROTO_ICMP)) == -1)
                err_sys("socket");

        bzero(&dstaddr, sizeof(dstaddr));
        dstaddr.sin_family = AF_INET;
        dstaddr.sin_port = htons(0);
        if (inet_pton(AF_INET, argv[1], &dstaddr.sin_addr) <= 0)
                err_sys("inet_pton");

        send_echo_req(sockfd, &dstaddr);
        recv_echo_reply(sockfd);

        exit(0);
}

void send_echo_req(int sockfd, struct sockaddr_in *dstaddr)
{
        char buf[100];
        size_t len = sizeof(struct icmp);
        struct icmp *icmp;
        socklen_t dstlen = sizeof(struct sockaddr_in);

        bzero(buf, sizeof(buf));
        icmp = (struct icmp *)buf;
        icmp->icmp_type = ICMP_ECHO;
        icmp->icmp_code = 0;
        icmp->icmp_id = getpid();
        icmp->icmp_seq = 1;
        icmp->icmp_cksum = in_cksum((uint16_t *) icmp, sizeof(struct icmp));

        if (sendto(sockfd, buf, len, 0, (SA *)dstaddr, dstlen) == -1)
                err_sys("sendto");
}

void recv_echo_reply(int sockfd)
{
        char buf[100];
        ssize_t n;
        struct ip *ip;
        struct icmp *icmp;
        while (1) {
                alarm(5); /* set timeout */
                if ((n = read(sockfd, buf, sizeof(buf))) == -1)
                        err_sys("read");

                ip = (struct ip *)buf;
                if (ip->ip_p != IPPROTO_ICMP) {
                        fprintf(stderr, "protocol error.\r\n");
                        exit(1);
                }
                icmp = (struct icmp *)(buf + sizeof(struct ip));
                if (icmp->icmp_type == ICMP_ECHOREPLY) {
                        if (icmp->icmp_id != getpid()) {
                                fprintf(stderr, "not this process.\r\n");
                                exit(1);
                        } else {
                                printf("destination host is alive.\r\n");
                                break;
                        }
                }
        }
}

uint16_t in_cksum(uint16_t *addr, int len)
{
        int nleft = len;
        uint32_t sum = 0;
        uint16_t *w = addr;
        uint16_t answer = 0;

        while (nleft > 1) {
                sum += *w++;
                nleft -= 2;
        }

        if (nleft == 1) {
                *(unsigned char *)(&answer) = *(unsigned char *)w ;
                sum += answer;
        }

        sum = (sum >> 16) + (sum & 0xffff);
        sum += (sum >> 16);
        answer = ~sum;
        return(answer);
}

void err_sys(const char *errmsg)
{
        perror(errmsg);
        exit(1);
}

收藏】 【评论】 【推荐】 【投稿】 【打印】 【关闭
发表评论
要记得去论坛讨论,点击注册新会员匿名评论
评论内容:不能超过250字,需审核后才会公布,请自觉遵守互联网相关政策法规。
阅读排行
随机推荐
实用信息推荐