【斯坦福CS144】Lab5

一、实验目的

在现有的NetworkInterface基础上实现一个IP路由器。

二、实验内容

在本实验中,你将在现有的NetworkInterface基础上实现一个IP路由器,从而结束本课程。路由器有几个网络接口,可以在其中任何一个接口上接收互联网数据报。路由器的工作是根据路由表转发它得到的数据报:一个规则列表,它告诉路由器,对于任何给定的数据报:

发送到哪个接口;

下一跳的IP地址 ;

你的工作是实现一个路由器,它可以为任何给定的数据报计算出这两件事。(你不需要实现设置路由表的算法,例如RIP、OSPF、BGP或SDN控制器,只需要实现跟随路由表的算法)。

你对路由器的实现将使用带有新的Router类的Sponge库,以及在模拟网络中检查你的路由器功能的测试。本实验建立在你在上一个实验中对NetworkInterface的实现之上,但不使用你在之前实验中实现的TCP栈。IP路由器不需要知道任何关于TCP、ARP或以太网的信息(仅限IP)。我们希望你的实现将需要大约25-30行的代码。

三、实验过程

在minnow目录下输入git merge origin/check5-startercode获取Lab5

用文本编辑器打开./src/router.hh

修改代码

用文本编辑器打开./src/router.cc

修改代码

在build目录下输入make进行编译

输入make check5进行测试

测试成功,实验结束

四、实验体会

1.本实验中的 Router 实现比较简单,只需实现一下 IP 最长匹配并将数据包转发即可。

2.需要注意的是,联系计算机网络理论课和实验所学习的内容,在实际网络中,路由表会根据网络拓扑和路由策略进行配置,以确保数据包能够正确地转发到目标。路由表中的路由条目根据目标网络地址的前缀匹配来确定数据包的转发规则。当无法找到匹配的路由条目时,数据包将根据默认路由进行转发,或者如果没有默认路由,则会被丢弃。

五、代码附录

router.hh

#pragma once#include "network_interface.hh"#include <optional>
#include <queue>// A wrapper for NetworkInterface that makes the host-side
// interface asynchronous: instead of returning received datagrams
// immediately (from the `recv_frame` method), it stores them for
// later retrieval. Otherwise, behaves identically to the underlying
// implementation of NetworkInterface.
class AsyncNetworkInterface : public NetworkInterface
{std::queue<InternetDatagram> datagrams_in_ {};public:using NetworkInterface::NetworkInterface;// Construct from a NetworkInterfaceexplicit AsyncNetworkInterface( NetworkInterface&& interface ) : NetworkInterface( interface ) {}// \brief Receives and Ethernet frame and responds appropriately.// - If type is IPv4, pushes to the `datagrams_out` queue for later retrieval by the owner.// - If type is ARP request, learn a mapping from the "sender" fields, and send an ARP reply.// - If type is ARP reply, learn a mapping from the "target" fields.//// \param[in] frame the incoming Ethernet framevoid recv_frame( const EthernetFrame& frame ){auto optional_dgram = NetworkInterface::recv_frame( frame );if ( optional_dgram.has_value() ) {datagrams_in_.push( std::move( optional_dgram.value() ) );}};// Access queue of Internet datagrams that have been receivedstd::optional<InternetDatagram> maybe_receive(){if ( datagrams_in_.empty() ) {return {};}InternetDatagram datagram = std::move( datagrams_in_.front() );datagrams_in_.pop();return datagram;}
};// A router that has multiple network interfaces and
// performs longest-prefix-match routing between them.
// class Router
// {
//   // The router's collection of network interfaces
//   std::vector<AsyncNetworkInterface> interfaces_ {};//   struct Route_entry//route_entry_
//   {
//     uint32_t route_prefix{};
//     uint8_t prefix_length{};
//     std::optional<Address> next_hop;
//     size_t interface_num{};
//   };
//   //static typedef struct route_entry_ Route_entry;
//   std::vector<Route_entry> route_table_{};
class Router
{// The router's collection of network interfacesstd::vector<AsyncNetworkInterface> interfaces_ {};struct Route_entry{uint32_t route_prefix {};uint8_t prefix_length {};std::optional<Address> next_hop;size_t interface_num {};};std::vector<Route_entry> route_table_ {};//std::vector<Route_entry>::iterator longest_prefix_match_( uint32_t dst_ip );//static int match_length_( uint32_t src_ip, uint32_t tgt_ip, uint8_t tgt_len );
public:// Add an interface to the router// interface: an already-constructed network interface// returns the index of the interface after it has been added to the routersize_t add_interface( AsyncNetworkInterface&& interface ){interfaces_.push_back( std::move( interface ) );return interfaces_.size() - 1;}// Access an interface by indexAsyncNetworkInterface& interface( size_t N ) { return interfaces_.at( N ); }// Add a route (a forwarding rule)void add_route( uint32_t route_prefix,uint8_t prefix_length,std::optional<Address> next_hop,size_t interface_num );// Route packets between the interfaces. For each interface, use the// maybe_receive() method to consume every incoming datagram and// send it on one of interfaces to the correct next hop. The router// chooses the outbound interface and next-hop as specified by the// route with the longest prefix_length that matches the datagram's// destination address.void route();std::vector<Route_entry>::iterator longest_prefix_match_( uint32_t dst_ip );int match_length_( uint32_t src_ip, uint32_t tgt_ip, uint8_t tgt_len );// std::vector<Route_entry>::iterator longest_prefix_match_( uint32_t dst_ip );// int match_length_( uint32_t src_ip, uint32_t tgt_ip, uint8_t tgt_len );
};

router.cc

#include "router.hh"#include <iostream>
#include <limits>
using namespace std;// route_prefix: The "up-to-32-bit" IPv4 address prefix to match the datagram's destination address against
// prefix_length: For this route to be applicable, how many high-order (most-significant) bits of
//    the route_prefix will need to match the corresponding bits of the datagram's destination address?
// next_hop: The IP address of the next hop. Will be empty if the network is directly attached to the router (in
//    which case, the next hop address should be the datagram's final destination).
// interface_num: The index of the interface to send the datagram out on.
// void Router::add_route( const uint32_t route_prefix,
//                         const uint8_t prefix_length,
//                         const optional<Address> next_hop,
//                         const size_t interface_num )
// {
//   cerr << "DEBUG: adding route " << Address::from_ipv4_numeric( route_prefix ).ip() << "/"
//        << static_cast<int>( prefix_length ) << " => " << ( next_hop.has_value() ? next_hop->ip() : "(direct)" )
//        << " on interface " << interface_num << "\n";//   // (void)route_prefix;
//   // (void)prefix_length;
//   // (void)next_hop;
//   // (void)interface_num;//   route_table_.emplace_back(route_prefix, prefix_length, next_hop, interface_num);
//   return;
// }// void Router::route()
// {
//   for ( auto& current_interface : interfaces_ ) {
//     auto received_dgram = current_interface.maybe_receive();
//     if ( received_dgram.has_value() ) {
//       auto& dgram = received_dgram.value();
//       if ( dgram.header.ttl > 1 ) {
//         dgram.header.ttl--;
//         // NOTE: important!!!
//         dgram.header.compute_checksum();
//         auto dst_ip = dgram.header.dst;
//         auto it = longest_prefix_match_( dst_ip );
//         if ( it != route_table_.end() ) {
//           auto& target_interface = interface( it->interface_num );
//           target_interface.send_datagram( dgram, it->next_hop.value_or( Address::from_ipv4_numeric( dst_ip ) ) );
//         }
//       }
//     }
//   }
// }// std::vector<Router::Route_entry>::iterator Router::longest_prefix_match_( uint32_t dst_ip )
// {
//   auto res = route_table_.end();
//   int max_length = 0;
//   for ( auto it = route_table_.begin(); it != route_table_.end(); ++it ) {
//     int len = match_length_( dst_ip, it->route_prefix, it->prefix_length );
//     if ( len > max_length ) {
//       max_length = len;
//       res = it;
//     }
//   }
//   return res;
// }// int Router::match_length_( uint32_t src_ip, uint32_t tgt_ip, uint8_t tgt_len )
// {
//   if ( tgt_len == 0 ) {
//     return 0;
//   }//   if ( tgt_len > 32 ) {
//     return -1;
//   }//   // tgt_len < 32
//   uint8_t const len = 32U - tgt_len;
//   src_ip = src_ip >> len;
//   tgt_ip = tgt_ip >> len;
//   return src_ip == tgt_ip ? tgt_len : -1;
// }void Router::add_route( const uint32_t route_prefix,const uint8_t prefix_length,const optional<Address> next_hop,const size_t interface_num )
{cerr << "DEBUG: adding route " << Address::from_ipv4_numeric( route_prefix ).ip() << "/"<< static_cast<int>( prefix_length ) << " => " << ( next_hop.has_value() ? next_hop->ip() : "(direct)" )<< " on interface " << interface_num << "\n";route_table_.emplace_back( route_prefix, prefix_length, next_hop, interface_num );
}void Router::route()
{for (uint32_t i=0; i<interfaces_.size();i++) {auto received_dgram = interface(i).maybe_receive();if ( received_dgram.has_value() ) {auto dgram = received_dgram.value();if ( dgram.header.ttl > 1 ) {dgram.header.ttl--;dgram.header.compute_checksum();auto dst_ip = dgram.header.dst;auto it = longest_prefix_match_( dst_ip );if ( it != route_table_.end() ) {//这里的buginterface( it->interface_num ).send_datagram( dgram, it->next_hop.value_or( Address::from_ipv4_numeric( dst_ip ) ) );}}}}
}std::vector<Router::Route_entry>::iterator Router::longest_prefix_match_( uint32_t dst_ip )
{auto res = route_table_.end();int max_length = -1;for ( auto it = route_table_.begin(); it != route_table_.end(); ++it ) {int len = match_length_( dst_ip, it->route_prefix, it->prefix_length );if ( len > max_length ) {max_length = len;res = it;}}return res;
}int Router::match_length_( uint32_t src_ip, uint32_t tgt_ip, uint8_t tgt_len )
{if ( tgt_len == 0 ) {return 0;}if ( tgt_len > 32 ) {return -1;}// tgt_len < 32uint8_t len = 32U - tgt_len;src_ip = src_ip >> len;tgt_ip = tgt_ip >> len;return src_ip == tgt_ip ? tgt_len : -1;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/881392.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【uniapp小程序】使用cheerio去除字符串中的HTML标签并获取纯文本内容

【uniapp小程序】使用cheerio去除字符串中的HTML标签并获取纯文本内容 参考资料安装引入使用 参考资料 【博主&#xff1a;AIpoem】uniapp小程序 使用cheerio处理网络请求拿到的dom数据 cheerio文档&#xff1a;https://github.com/cheeriojs/cheerio/wiki/Chinese-README 安…

Sequelize 做登录查询数据

在 Sequelize 中处理登录请求通常意味着你需要根据提供的用户名或电子邮件以及密码来查询数据库中的用户。由于密码在数据库中应该是以哈希形式存储的&#xff0c;因此你还需要验证提供的密码是否与存储的哈希密码匹配。 以下是一个简单的例子&#xff0c;展示了如何使用 Sequ…

SpringBoot美发门店系统:提升服务质量

摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了美发门店管理系统的开发全过程。通过分析美发门店管理系统管理的不足&#xff0c;创建了一个计算机管理美发门店管理系统的方案。文章介绍了美发门店管理系统的系…

SpringBoot访问web中的静态资源

SpringBoot访问web中的静态资源&#xff0c;有两个方式&#xff1a; 1、SpringBoot默认指定了一些固定的目录结构&#xff0c;静态资源放到这些目录中的某一个&#xff0c;系统运行后浏览器就可以访问到 ① 关键是SpringBoot默认指定的可以存放静态资源的目录有哪些&#xff…

U mamba配置问题;‘KeyError: ‘file_ending‘

这个错误仍然是因为在 dataset_json 中找不到 file_ending 键。请尝试以下步骤&#xff1a; 检查 JSON 文件&#xff1a;确认 JSON 文件中确实有 file_ending&#xff0c;并且它的拼写完全正确。 打印 JSON 内容&#xff1a;在抛出异常之前&#xff0c;添加打印语句输出 datas…

JavaScript 数组简单学习

目录 1. 数组 1.1 介绍 1.2 基本使用 1.2.1 声明语法 1.2.2 取值语法 1.2.3 术语 1.3 案例 1. 数组 1.1 介绍 1.2 基本使用 1.2.1 声明语法 1.2.2 取值语法 1.2.3 术语 1.3 案例

Python知识点:如何应用Python工具,使用NLTK进行语言模型构建

开篇&#xff0c;先说一个好消息&#xff0c;截止到2025年1月1日前&#xff0c;翻到文末找到我&#xff0c;赠送定制版的开题报告和任务书&#xff0c;先到先得&#xff01;过期不候&#xff01; 如何使用NLTK进行语言模型构建 在自然语言处理&#xff08;NLP&#xff09;中&a…

pikachu靶场总结(三)

五、RCE 1.RCE(remote command/code execute)概述 RCE漏洞&#xff0c;可以让攻击者直接向后台服务器远程注入操作系统命令或者代码&#xff0c;从而控制后台系统。 远程系统命令执行 一般出现这种漏洞&#xff0c;是因为应用系统从设计上需要给用户提供指定的远程命令操作的…

基于SpringBoot和Vue的餐饮管理系统

基于springbootvue实现的餐饮管理系统 &#xff08;源码L文ppt&#xff09;4-078 第4章 系统设计 4.1 总体功能设计 一般个人用户和管理者都需要登录才能进入餐饮管理系统&#xff0c;使用者登录时会在后台判断使用的权限类型&#xff0c;包括一般使用者和管理者,一…

星融元P4交换机:在全球芯片短缺中,为您的网络可编程之路保驾护航

当数字化转型成为新常态&#xff0c;云计算、物联网、5G和人工智能等技术正以惊人的速度进步&#xff0c;重塑了我们对网络设备性能和适应性的预期。在这场技术革新的浪潮中&#xff0c;网络的灵活性、开放性和编程能力成为了推动行业发展的关键。P4可编程交换机&#xff0c;以…

飞驰云联入围2024西门子Xcelerator公开赛50强

近日&#xff0c;备受瞩目的西门子 Xcelerator公开赛公布结果&#xff0c;经过激烈的筛选&#xff0c;Ftrans飞驰云联《Ftrans制造业数据交换安全管控解决方案》凭借优异的表现&#xff0c;成功入围 Xcelerator公开赛50强&#xff01; Xcelerator 公开赛以工信部智能制造典型场…

胤娲科技:00后揭秘——AI大模型的可靠性迷局

当智能不再“靠谱”&#xff0c;我们该何去何从&#xff1f; 想象一下&#xff0c;你向最新的GPT模型提问&#xff1a;“9.9和9.11哪个大&#xff1f;”这本应是个小菜一碟的问题&#xff0c;却足以让不少高科技的“大脑”陷入沉思&#xff0c; 甚至给出令人啼笑皆非的答案。近…

实战逆向RUST语言程序

实战为主&#xff0c;近日2024年羊城杯出了一道Rust编写的题目&#xff0c;这里将会以此题目为例&#xff0c;演示Rust逆向该如何去做。 题目名称&#xff1a;sedRust_happyVm 题目内容&#xff1a;unhappy rust, happy vm 关于Rust逆向&#xff0c;其实就是看汇编&#xff…

太阳诱电电感选型方法及产品介绍

功率电感在电子电路中被广泛应用&#xff0c;太阳诱电的功率电感从原材料开始进行研发&#xff0c;生产和销售。 本次研讨会将带领大家更加了解功率电感的选型方法&#xff0c;以及各种功率电感的种类和特征。 此外&#xff0c;也将介绍太阳诱电的最新产品阵容。本次研讨会预计…

边学边用docker-为什么要进到容器里面修改权限

在 Docker 容器中修改文件夹权限&#xff0c;通常需要进入容器内部来执行命令&#xff0c;这是因为 Docker 容器提供了一个隔离的环境&#xff0c;其内部的文件系统与宿主机是隔离的。 1. 隔离性&#xff1a;Docker 容器设计为轻量级的隔离环境&#xff0c;每个容器都有自己的…

如何在表中设置主键约束?

主键约束是数据库表设计中的一种完整性约束&#xff0c;用于唯一标识表中的每一行。 每个表只能有一个主键&#xff0c;且主键值必须是唯一的&#xff0c;并且不能为NULL。主键可以由一个字段或多个字段组成。 主键的存在有助于提高数据检索的速度和确保数据的完整性和一致性…

vue2和vue3中的组件间通信知识点总结

vue2中组件间通信 在Vue2中&#xff0c;组件间的通信是构建复杂应用的关键。以下是一些常见的Vue2组件间通信方式&#xff0c;并附有详细说明和示例&#xff1a; 1. Props&#xff08;父子组件通信&#xff09; Props是Vue中用于父子组件通信的一种机制。父组件通过属性&…

社工字典生成工具 —— CeWL 使用手册

GitHub - digininja/CeWL: CeWL is a Custom Word List GeneratorCeWL is a Custom Word List Generator. Contribute to digininja/CeWL development by creating an account on GitHub.https://github.com/digininja/CeWL/ 0x01&#xff1a;CeWL 简介 CeWL&#xff08;Cust…

openmmlab实现图像超分辨率重构

文章目录 前言一、图像超分辨率重构简介二、mmmagic实现图像超分 前言 超分辨率重构技术&#xff0c;作为计算机视觉领域的一项重要研究课题&#xff0c;近年来受到了广泛关注。随着科技的飞速发展&#xff0c;人们对图像质量的要求越来越高&#xff0c;尤其是在智能手机、监控…

【算法】链表:24.两两交换链表中的节点

目录 1、题目链接 2、题目介绍 3、解法 4、代码 1、题目链接 24. 两两交换链表中的节点 - 力扣&#xff08;LeetCode&#xff09; 2、题目介绍 3、解法 引入伪头节点&#xff1a; 为了处理头节点可能被交换的情况&#xff0c;我们引入一个伪头节点&#xff08;dummy no…