babyos2(39) network(5) 您所在的位置:网站首页 echo和echo-reply babyos2(39) network(5)

babyos2(39) network(5)

2024-07-09 10:14| 来源: 网络整理| 查看: 265

前面为babyos2 实现了发送IP数据报,但只测试了同一个局域网内发送,而当目的IP跟发送发自己的IP不在同一局域网内时,babyos2会把IP数据报发往网关。而如何测试是否发送成功,首先想到的是ping。ping使用ICMP(Internet控制报文协议)。

ICMP经常被认为是 IP层的一个组成部分。它传递差错报文以及其他需要注意的信息。ICMP报文通常被 IP层或更高层协议( TCP或UDP)使用。一些 ICMP报文把差错报文返回给用户进程。

ICMP报文有多种类型,如请求回显,回显应答,目的不可达,源端被关闭,重定向,超时,地址掩码请求等。我们这里只关注请求回显和回显应答。

目前从用户态发起ping请求时机还不成熟(需要socket),暂时让babyos2从内核态发送ICMP报文,当收到请求时发送reply,及接收reply等。

1. ICMP 回显请求和回显应答报文格式

class icmp_echo_hdr_t { public: void init(uint8 type, uint8 code, uint16 check_sum, uint16 id, uint16 seq); public: uint8 m_type; uint8 m_code; uint16 m_check_sum; uint16 m_id; uint16 m_seq_no; }; void icmp_echo_hdr_t::init(uint8 type, uint8 code, uint16 check_sum, uint16 id, uint16 seq) {     m_type      = type;     m_code      = code;     m_check_sum = check_sum;     m_id        = id;     m_seq_no    = seq; }

2. ICMP echo request

bool icmp_t::echo_request(uint32 ip, uint16 id, uint16 seq, uint8* data, uint32 len) { uint32 total = len + sizeof(icmp_echo_hdr_t); net_buf_t* buffer = os()->get_net()->alloc_net_buffer(total); if (buffer == NULL) { console()->kprintf(RED, "ICMP echo_request, alloc net buffer failed, total: %u\n", total); return false; } console()->kprintf(GREEN, "send an icmp echo request to ip: "); net_t::dump_ip_addr(ip); console()->kprintf(GREEN, " seq: %u\n", seq); icmp_echo_hdr_t hdr; hdr.init(ECHO_REQUEST, /* type */ 0, /* code */ 0, /* check sum */ net_t::htons(id), /* id */ net_t::htons(seq)); /* seq no */ hdr.m_check_sum = net_t::check_sum((uint8 *) &hdr, sizeof(icmp_echo_hdr_t)); buffer->m_data = (uint8 *) buffer + sizeof(net_buf_t); buffer->m_data_len = total; uint8* p = buffer->m_data; memcpy(p, &hdr, sizeof(icmp_echo_hdr_t)); if (data != NULL && len != 0) { p += sizeof(icmp_echo_hdr_t); memcpy(p, data, len); } os()->get_net()->get_ip()->transmit(ip, buffer->m_data, total, ip_t::PROTO_ICMP); os()->get_net()->free_net_buffer(buffer); return true; }

代码比较简单,初始化一个报文头,

ip表示向哪个地址发送ping请求;

id是为了当系统中有多个ping同时发起请求时,当收到回复时识别是回复的那个ping的请求,一般会传进程ID,这里我们暂时只从内核发出请求,所以传0;

seq表示序列号,一般是一个递增的数字,表示序列;

data表示发送的数据,暂时不发任何数据,len为0.

准备号头后,计算校验和,然后分配一个buffer用于存储信息,最后发出去。

2. ICMP echo receive

receive表示收到一个ICMP报文,我们首先判断它的类型,根据类型做处理

void icmp_t::receive(net_buf_t* buf, uint32 ip) { uint8 type = *(uint8 *) buf->m_data; switch (type) { case ECHO_REQUEST: echo_request_receive(buf, ip); break; case ECHO_REPLY: echo_reply_receive(buf, ip); break; default: console()->kprintf(RED, "receive an icmp package, but not support the type %x now.\n", type); break; } }

暂时只支持 ECHO_REQUEST跟ECHO_REPLY,分别表示收到一个ICMP echo请求跟答复。

void icmp_t::echo_request_receive(net_buf_t* buf, uint32 ip) {     icmp_echo_hdr_t* hdr = (icmp_echo_hdr_t *) buf->get_data();     uint16 check_sum = net_t::check_sum(buf->get_data(), buf->get_data_len());     if (check_sum != 0) {         console()->kprintf(RED, "receive an icmp echo request, but checksum is error: %x.\n", check_sum);         return;     }     console()->kprintf(GREEN, "receive an icmp echo request from ip: ");     net_t::dump_ip_addr(ip);     console()->kprintf(GREEN, " seq: %u\n", net_t::ntohs(hdr->m_seq_no));     echo_reply(ip, net_t::ntohs(hdr->m_id), net_t::ntohs(hdr->m_seq_no), NULL, 0); } void icmp_t::echo_reply_receive(net_buf_t* buf, uint32 ip) {     icmp_echo_hdr_t* hdr = (icmp_echo_hdr_t *) buf->get_data();     uint16 check_sum = net_t::check_sum((uint8 *) hdr, sizeof(icmp_echo_hdr_t));     if (check_sum != 0) {         console()->kprintf(RED, "receive an icmp echo reply, but checksum is error: %x.\n", check_sum);         return;     }     console()->kprintf(WHITE, "receive an icmp echo reply from ip: ");     net_t::dump_ip_addr(ip);     console()->kprintf(WHITE, " seq: %u\n", net_t::ntohs(hdr->m_seq_no)); }

当收到一个请求时,需要答复,当收到一个答复时只简单打印一些信息表示收到答复。

3.ICMP echo reply

bool icmp_t::echo_reply(uint32 ip, uint16 id, uint16 seq, uint8* data, uint32 len) { uint32 total = len + sizeof(icmp_echo_hdr_t); net_buf_t* buffer = os()->get_net()->alloc_net_buffer(total); if (buffer == NULL) { console()->kprintf(RED, "ICMP echo_reply, alloc net buffer failed, total: %u\n", total); return false; } console()->kprintf(GREEN, "send an icmp echo reply to ip: "); net_t::dump_ip_addr(ip); console()->kprintf(GREEN, " seq: %u\n", seq); icmp_echo_hdr_t hdr; hdr.init(ECHO_REPLY, /* type */ 0, /* code */ 0, /* check sum */ net_t::htons(id), /* id */ net_t::htons(seq)); /* seq no */ hdr.m_check_sum = net_t::check_sum((uint8 *) &hdr, sizeof(icmp_echo_hdr_t)); buffer->m_data = (uint8 *) buffer + sizeof(net_buf_t); buffer->m_data_len = total; uint8* p = buffer->m_data; memcpy(p, &hdr, sizeof(icmp_echo_hdr_t)); if (data != NULL && len != 0) { p += sizeof(icmp_echo_hdr_t); memcpy(p, data, len); } os()->get_net()->get_ip()->transmit(ip, buffer->m_data, total, ip_t::PROTO_ICMP); os()->get_net()->free_net_buffer(buffer); return true; }

答复跟请求类似,代码比较简单,不做过多解释。

4.测试

上图是启动两个虚拟机,其中一个向另一个发送ICMP echo请求,可以发现104向105发送请求,105收到请求并作出答复,104收到答复。

这是wireshark抓包的结果。

下一步是测试ping baidu,因为babyos2暂时不支持DNS,所以先在linux上ping baid.com得到它的IP,再让babyos2向这个IP发送ICMP echo请求:

可以发现向111.13.101.208发送ICMP echo request,因为该IP不在同一局域网内,将数据包发送到了网关,IP 182.168.1.1,然后收到了来自111.13.101.208的ICMP echo reply。

从抓到的包来看,确实收到了来自baidu的回显答复。

-----------------------------------------------------------------------------------------------------

到目前为止babyos2具有了简单的往某个IP发送数据报的能力,此时实现socket IF_INET的时机已基本成熟,前面实现过通过socket进行进程间通信,通过网络与之相似但要复杂一些,特别是有连接的SOCK_STREAM 需要实现TCP。但无连接的SOCK_DGRAM 跟SOCK_RAW已经能够实现。

后面的计划:

1)socket SOCK_RAW, 测试方式是用户进程ping,通过SOCK_RAW发送ICMP echo请求

2)socket SOCK_DGRAM, 需要先支持UDP

3)socket SOCK_STREAM,需要支持TCP



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有