OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 9211|回复: 12

让LwIP拥有PING其他设备的能力

[复制链接]

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
发表于 2016-8-2 18:22:04 | 显示全部楼层 |阅读模式
10金钱
请问有没有人使用过LWIP(1.4.1版本)中的ICMP协议实现设备PING PC机或其他设备的测试,有得话请指教,最好可以附上例程,感激不尽。。。
求助。。。

最佳答案

查看完整内容[请看2#楼]

太久没回来看帖,帐号直接被冻结。 其实机制很简单,ICMP其实基于IP层的,发送的时候只需要根据ICMP包的协议自己组一个一样的协议包就可以。至于时间的计算就是在发送的时候先获取一下单片机的运行时间,然后在ICMP Respond解析那里再一次获取一下单片机的运行时间,两个时间一减就是PING包的时间,不过肯定是有点延时的,但现在用起来其实还可以。 上码,不BB。 ICMP发送: err_t udp_sendicmp(struct udp_pcb *pcb, struct ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
 楼主| 发表于 2016-8-2 18:22:05 | 显示全部楼层
太久没回来看帖,帐号直接被冻结。
其实机制很简单,ICMP其实基于IP层的,发送的时候只需要根据ICMP包的协议自己组一个一样的协议包就可以。至于时间的计算就是在发送的时候先获取一下单片机的运行时间,然后在ICMP Respond解析那里再一次获取一下单片机的运行时间,两个时间一减就是PING包的时间,不过肯定是有点延时的,但现在用起来其实还可以。
上码,不BB。
ICMP发送:
err_t
udp_sendicmp(struct udp_pcb *pcb, struct pbuf *p,
  struct ip_addr *dst_ip, u16_t dst_port)
{
  struct netif *netif;
  LWIP_DEBUGF(UDP_DEBUG | LWIP_DBG_TRACE | 3, ("udp_send\n"));
  /* find the outgoing network interface for this packet */
#if LWIP_IGMP
  netif = ip_route((ip_addr_ismulticast(dst_ip))?(&(pcb->multicast_ip))dst_ip));
#else
  netif = ip_route(dst_ip);
#endif /* LWIP_IGMP */
  /* no outgoing network interface could be found? */
  if (netif == NULL) {
    LWIP_DEBUGF(UDP_DEBUG | 1, ("udp_send: No route to 0x%"X32_F"\n", dst_ip->addr));
    UDP_STATS_INC(udp.rterr);
    return ERR_RTE;
  }
  return udp_sendto_ping(pcb, p, dst_ip, dst_port, netif);
}

err_t
udp_sendto_ping(struct udp_pcb *pcb, struct pbuf *p,
  struct ip_addr *dst_ip, u16_t dst_port, struct netif *netif)
{
        //  struct udp_hdr *udphdr;
        struct ip_addr *src_ip;
        err_t err;
        struct pbuf *q; /* q will be sent down the stack */
        struct icmp_echo_hdr *icmphdr; //kaishui

        /* not enough space to add an UDP header to first pbuf in given p chain? */
        if (pbuf_header(p, sizeof(struct icmp_echo_hdr))) {
                /* allocate header in a separate new pbuf */
                //        q = pbuf_alloc(PBUF_IP, sizeof(struct icmp_echo_hdr), PBUF_RAM);
                q = pbuf_alloc(PBUF_IP, sizeof(struct icmp_echo_hdr)+ IP_HLEN + ICMP_DEST_UNREACH_DATASIZE,
                        PBUF_RAM);
                /* new header pbuf could not be allocated? */
                /* chain header q in front of given pbuf p */
                pbuf_chain(q, p);
                /* first pbuf q points to header pbuf */
        }
  
        /* q now represents the packet to be sent */
        icmphdr = (struct icmp_echo_hdr *)q->payload;
        icmphdr->type = 8;
        icmphdr->code = 0;
        icmphdr->id =htons(0x0100)  ;     
        icmphdr->seqno =htons(0x51d1);
        icmphdr->chksum = 0;  
        //   icmphdr->chksum = inet_chksum(p->payload, 10);
        icmphdr->chksum = inet_chksum(icmphdr, q->len);
        /* PCB local address is IP_ANY_ADDR? */
    if (ip_addr_isany(&pcb->local_ip))
        {
                /* use outgoing network interface IP address as source address */
                src_ip = &(netif->ip_addr);
    }
        else
        {
                /* check if UDP PCB local IP address is correct
                 * this could be an old address if netif->ip_addr has changed */
                /* use UDP PCB local IP address as source address */
                src_ip = &(pcb->local_ip);
        }
        err = ip_output_if(q, src_ip, dst_ip, pcb->ttl, pcb->tos, IP_PROTO_ICMP, netif);
        snmp_inc_udpoutdatagrams();
        /* did we chain a separate header pbuf earlier? */
        if (q != p)
        {
                /* free the header pbuf */
                pbuf_free(q);
                q = NULL;
                /* p is still referenced by the caller, and will live on */
        }
        UDP_STATS_INC(udp.xmit);
        pbuf_free(p);//important
        return err;
}
void PingInit(void)
{
        struct ip_addr ipaddr;
        unsigned int port = 2000; //69
        p = pbuf_alloc(PBUF_RAW,10,PBUF_RAM);
        p->payload=(void *)UDPData;
        IP4_ADDR(&ipaddr,0,0,0,0);  
        UDPpcb = udp_new();
        udp_bind(UDPpcb,&ipaddr, port);
}

void ICMP_Ping(void)
{
        struct ip_addr dstipaddr;
               
        p = pbuf_alloc(PBUF_RAW,10,PBUF_RAM);
        p->payload=(void *)UDPData;
               
        OSTimeDlyHMSM(0, 0, 0, 500);        //wait 200ms
        memcpy(&dstipaddr.addr,&flash_read.lwip_msg->dns2,4);               
       
        if(mt_client.respond_flag ==0)
        {
                cnt_1ms = 0;
        }
        else
        {
                mt_client.ping_end.end_time.tv_usec = 1000;
                mt_client.timeout_flag = 1;
        }
        udp_sendicmp(UDPpcb,p,&dstipaddr,2000);
        mt_client.respond_flag = 1;
        ntx_packet++;
        mt_client.ping_before.end_time.tv_sec = get_running_time();
        mt_client.ping_before.end_time.tv_usec = get_1mstime();//ms         
}

ICMP接收:(icmp.c)
void
icmp_input(struct pbuf *p, struct netif *inp)
{
   ....
     case ICMP_ER:
    /* This is OK, echo reply might have been parsed by a raw PCB
       (as obviously, an echo request has been sent, too). */
#ifdef PINGPACK_TEST               
        nrx_packet ++;
        mt_client.ping_after.end_time.tv_sec = get_running_time();
        mt_client.ping_after.end_time.tv_usec = get_1mstime();//ms
        mt_client.ping_end.end_time.tv_sec = 0;
        mt_client.ping_end.end_time.tv_usec = mt_client.ping_after.end_time.tv_usec - mt_client.ping_before.end_time.tv_usec;
        mt_client.respond_flag = 0;
        if(mt_client.ping_end.end_time.tv_usec > 10)//ms
        {
                mt_client.timeout_flag = 1;
        }
#endif  
    break;
    ...
回复

使用道具 举报

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
 楼主| 发表于 2016-8-2 18:23:24 | 显示全部楼层

@原子哥速来求解,感谢!!!
回复

使用道具 举报

558

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
164897
金钱
164897
注册时间
2010-12-1
在线时间
2100 小时
发表于 2016-8-3 21:16:12 | 显示全部楼层
这个功能没用过哦,帮顶.
回复

使用道具 举报

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
 楼主| 发表于 2016-8-4 14:54:03 | 显示全部楼层
正点原子 发表于 2016-8-3 21:16
这个功能没用过哦,帮顶.

已经通过UDP的方式发送ping包了,但是要取得respond的延时时间,还在搞
回复

使用道具 举报

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
 楼主| 发表于 2016-8-4 14:54:15 | 显示全部楼层
正点原子 发表于 2016-8-3 21:16
这个功能没用过哦,帮顶.

谢谢原子哥
回复

使用道具 举报

17

主题

237

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1312
金钱
1312
注册时间
2017-3-1
在线时间
259 小时
发表于 2017-8-16 15:45:56 | 显示全部楼层
楼主你这后面还有下文吗?搞出来没?
回复

使用道具 举报

5

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
79
金钱
79
注册时间
2015-5-26
在线时间
10 小时
 楼主| 发表于 2017-12-18 15:55:37 | 显示全部楼层
NewGuard 发表于 2017-8-16 15:45
楼主你这后面还有下文吗?搞出来没?

搞出来了
回复

使用道具 举报

14

主题

54

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
204
金钱
204
注册时间
2017-12-4
在线时间
62 小时
发表于 2018-8-30 11:53:53 | 显示全部楼层

正好项目需要,可以发一份完整的源码给我吗,谢谢你 516182660@qq.com
回复

使用道具 举报

14

主题

204

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2443
金钱
2443
注册时间
2017-9-13
在线时间
559 小时
发表于 2018-8-31 11:08:52 | 显示全部楼层
请问楼主对端收到udp数据包后能解析出来是icmp报文?
回复

使用道具 举报

0

主题

91

帖子

0

精华

初级会员

Rank: 2

积分
177
金钱
177
注册时间
2018-9-27
在线时间
15 小时
发表于 2018-10-11 09:53:16 | 显示全部楼层
还有这种操作,666
回复

使用道具 举报

0

主题

1

帖子

0

精华

新手入门

积分
2
金钱
2
注册时间
2019-8-2
在线时间
0 小时
发表于 2019-8-2 19:30:02 | 显示全部楼层
能给个这个ping的代码吗? 38384298@qq.com
回复

使用道具 举报

4

主题

18

帖子

0

精华

初级会员

Rank: 2

积分
85
金钱
85
注册时间
2021-8-2
在线时间
11 小时
发表于 2021-8-30 15:58:06 | 显示全部楼层
新手求源码 liu_zx1213@163.com   
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2024-6-9 06:39

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表