jjzjj

CS144-Lab6

MINGgoS 2023-08-09 原文

概述

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

  • 发送到哪个接口;
  • 下一跳的IP地址 ;

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

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

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-3DONudSr-1677328903313)(null)]

图1:路由器包含多个网络接口,可以在其中任何一个接口上接收IP数据报。路由器将接收到的任何数据报转发到相应出站接口上的下一跳,路由表告诉路由器如何做出这个决定。

开始

  1. 请确保你已经提交了你在实验5中的所有解决方案。请不要修改libsponge目录顶层以外的任何文件,或者webget.cc。(请不要添加代码所依赖的额外文件。)否则,你可能会在合并实验6的启动代码时遇到麻烦。
  2. 在实验作业的存储库中,运行git fetch来检索实验作业的最新版本。
  3. 通过运行git merge origin/lab6-startercode,下载实验6的启动代码。
  4. build目录中,编译源代码:make(编译时可以运行make -j4以使用四个处理器)。
  5. build目录外,打开并开始编辑writeups/lab6.md文件。这是你实验报告的模板,将包含在你提交的内容中。

实现路由器

在本实验中,你将实现一个Router类,它可以:

  • 跟踪路由表(转发规则或路由列表),并
  • 转发它收到的每个数据报:
    • 转发到正确的下一跳
    • 在正确的出站NetworkInterface

你的实现将被添加到router.hhrouter.cc骨架文件中。在你开始编码之前,请查看新的Router类的文档

下面是你要实现的两个方法,以及我们对每个方法的期望:

void add_route(const uint32_t route_prefix,
               const uint8_t prefix_length,
               const optional<Address> next_hop,
               const size_t interface_num);

这个方法将一条路由添加到路由表中。你要在Router类中添加一个数据结构作为私有成员来存储这些信息。这个方法所要做的就是保存路由,以供以后使用。

路由的各个部分是什么意思?

路由是一个”匹配——行动”规则:它告诉路由器,如果一个数据报前往一个特定的网络(一个IP地址范围),并且如果该路由被选为最具体的匹配路由,那么路由器应该把数据报转发到特定接口上的特定下一跳。

“匹配”:数据报是前往这个网络的吗?route_prefixprefix_length共同指定了一个可能包括数据报目的地的IP地址范围(一个网络)。route_prefix是一个32位数字的IP地址。prefix_length是一个介于0和32(包括32)之间的数字;它告诉路由器路由前缀中有多少最高有效位是有效的。例如,要表达一个到网络”18.47.0.0/16”的路由(这与前两个字节为18和47的任何32位IP地址匹配),路由前缀将是305070080(18×224+47×216),前缀长度是16。任何以”18.47.x.y”为目的地的数据报都会匹配。

**“行动”:如果路由匹配并被选中,该怎么做。**如果路由器直接连接到有关的网络,next_hop将是一个空的可选项;在这种情况下,next_hop是数据报的目标地址。但如果路由器是通过其他路由器连接到有关网络的,则next_hop将包含路径中下一路由器的IP地址。interface_num给出了路由器NetworkInterface的索引,它用来将数据报发送到下一跳。你可以用interface(interface_num)方法访问这个接口。

void route_one_datagram(InternetDatagram &dgram);

这里是橡胶与道路的交汇处。这个方法需要将数据报路由到下一跳,从适当的接口传出。它需要实现IP路由器的”最长前缀匹配”逻辑,以找到最佳路由,这意味着:

  • 路由器搜索路由表,以找到与数据报的目的地址相匹配的路由。我们所说的”匹配”是指目的地址的最高有效prefix_length比特与route_prefix的最高有效prefix_length比特相同的。
  • 在匹配的路由中,路由器选择具有最大prefix_length的路由,这就是最长前缀匹配路由。
  • 如果没有匹配的路由,路由器会丢弃数据报。
  • 路由器会递减数据报的TTL(生存时间)。如果TTL已经为零,或在递减后为零,路由器应该放弃该数据报。
  • 否则,路由器将修改后的数据报通过适当的接口(interface(interface_num).send_datagram())发送到适当的下一跳。

在这个互联网的设计中,有个优点(或至少是一种成功的抽象):路由器从不考虑TCP、ARP或以太网帧。路由器甚至不知道链路层是什么样子的。路由器只考虑互联网数据包,并且只通过NetworkInterface抽象与链路层进行交互。当涉及到”链路层地址是如何解决的?”或”链路层是否有自己的不同于IP的寻址方案?”或”链路层帧的格式是什么?”或”数据报的有效载荷是什么意思?”等问题时,路由器根本不关心。

//! \brief 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_out{};

  public:
    using NetworkInterface::NetworkInterface;

    //! Construct from a NetworkInterface
    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 frame
    void recv_frame(const EthernetFrame &frame) {
        auto optional_dgram = NetworkInterface::recv_frame(frame);
        if (optional_dgram.has_value()) {
            _datagrams_out.push(std::move(optional_dgram.value()));
        }
    };

    //! Access queue of Internet datagrams that have been received
    std::queue<InternetDatagram> &datagrams_out() { return _datagrams_out; }
};

struct route_rule{
    uint32_t route_prefix{};
    uint8_t prefix_length{};
    std::optional<Address> next_hop;
    size_t interface_num{};
    route_rule(const uint32_t _route_prefix,
                const uint8_t _prefix_length,
                const std::optional<Address> _next_hop,
                const size_t _interface_num)
    :route_prefix(_route_prefix),prefix_length(_prefix_length)
    ,next_hop(_next_hop),interface_num(_interface_num)
    {}
};

//! \brief 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{};

    //! Send a single datagram from the appropriate outbound interface to the next hop,
    //! as specified by the route with the longest prefix_length that matches the
    //! datagram's destination address.
    void route_one_datagram(InternetDatagram &dgram);

    std::vector<route_rule> _rules{};

  public:
    //! Add an interface to the router
    //! \param[in] interface an already-constructed network interface
    //! \returns The index of the interface after it has been added to the router
    size_t add_interface(AsyncNetworkInterface &&interface) {
        _interfaces.push_back(std::move(interface));
        return _interfaces.size() - 1;
    }

    //! Access an interface by index
    AsyncNetworkInterface &interface(const size_t N) { return _interfaces.at(N); }

    //! Add a route (a forwarding rule)
    void add_route(const uint32_t route_prefix,
                   const uint8_t prefix_length,
                   const std::optional<Address> next_hop,
                   const size_t interface_num);

    //! Route packets between the interfaces
    void route();
    
    bool prefix_equal(uint32_t ip1, uint32_t ip2, uint8_t prefix_length);
};

这段代码实现了一个多接口的路由器,其中每个接口都有一个异步接收数据报的队列。路由器的功能是根据最长前缀匹配算法对接收到的每个数据报进行路由,并将其发送到下一跳,以使其达到目标地址。

AsyncNetworkInterface是一个对NetworkInterface的封装,它具有一个异步队列_datagrams_out,在接收到数据报后将其推送到队列中。它还覆盖了NetworkInterfacerecv_frame方法,以便将数据报推送到队列中,以供稍后检索。

Router类有一个包含所有网络接口的向量 _interfaces,一个包含所有路由表规则的向量 _rules,以及一个add_interface方法,用于添加接口。路由器还实现了一个add_route方法,该方法接受路由前缀、路由前缀长度、下一跳IP地址(可选)以及出站接口的索引。它将所有路由规则存储在一个向量中,以便在进行数据报路由时进行查找。此外,还实现了一个route方法,用于将路由器的所有接口一起轮询,处理每个接口上接收到的数据报,并调用route_one_datagram方法将其路由到下一跳。最后,还实现了一个prefix_equal方法,用于检查两个IP地址的前缀是否相同。

在路由表的最长前缀匹配算法中,对于目的地址,路由器从路由表中选择最具体的路由,该路由匹配前缀长度最长,而且其匹配的前缀位数与目标地址的前缀位数相同。prefix_equal方法用于比较两个IP地址的前缀。

//! \param[in] route_prefix The "up-to-32-bit" IPv4 address prefix to match the datagram's destination address against
//! \param[in] 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?
//! \param[in] 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).
//! \param[in] 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() << "/" << int(prefix_length)
         << " => " << (next_hop.has_value() ? next_hop->ip() : "(direct)") << " on interface " << interface_num << "\n";
    route_rule r(route_prefix,prefix_length,next_hop,interface_num);
    _rules.push_back(move(r));
}

Router::add_route 函数添加一个路由规则,包含路由表项中的路由前缀,前缀长度,下一跳地址(如果有),以及数据包的输出接口索引。这个函数打印了调试信息,用于检查添加的路由规则。

//! \param[in] dgram The datagram to be routed
void Router::route_one_datagram(InternetDatagram &dgram) {
    uint32_t dst = dgram.header().dst;
    optional<Address> next_hop{};
    size_t interface_num{};
    optional<uint8_t> best_by_far{};

    for(auto& r: _rules) {
        if((!best_by_far.has_value() || r.prefix_length>best_by_far.value()) && prefix_equal(dst,r.route_prefix,r.prefix_length)){
            next_hop = r.next_hop;
            interface_num = r.interface_num;
            best_by_far = r.prefix_length;
        }
    }

    if(best_by_far.has_value()){
        if(dgram.header().ttl>1) {
            dgram.header().ttl--;

            if(next_hop.has_value()) {
                interface(interface_num).send_datagram(dgram,next_hop.value());
            } 
            // if not have next_hop, the next_hop is dgram's destination hop
            else {
                interface(interface_num).send_datagram(dgram,Address::from_ipv4_numeric(dst));
            }        
        }
    }
    
    return;
}

Router::route_one_datagram 函数根据路由表将数据报路由到下一个合适的接口。它使用路由表中最长前缀匹配的规则来查找应该用哪个输出接口。如果找到合适的路由规则,则检查数据报的 TTL 是否大于1。如果是,减少TTL并将数据报发送到下一个合适的接口。如果找不到合适的路由规则,数据报将被丢弃。

void Router::route() {
    // Go through all the interfaces, and route every incoming datagram to its proper outgoing interface.
    for (auto &interface : _interfaces) {
        auto &queue = interface.datagrams_out();
        while (not queue.empty()) {
            route_one_datagram(queue.front());
            queue.pop();
        }
    }
}

Router::route 函数遍历路由器的所有网络接口,将每个接口的入队数据报依次路由到正确的输出接口上。

bool Router::prefix_equal(uint32_t ip1, uint32_t ip2, uint8_t prefix_length) {
    if(prefix_length==0){
        return true;
    } else {
        return (ip1 >> (32-prefix_length)) == (ip2 >> (32-prefix_length));
    }
}

Router::prefix_equal 函数用于比较两个IP地址的最高 prefix_length 位是否相等。如果前缀长度为0,将返回 true。如果前缀长度不为0,则将两个地址右移(32-prefix_length)位,并比较结果是否相等。

测试

你可以通过运行make checklab6来测试你的实现。这将在特定的模拟网络中测试路由器,如图2所示。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RIREzhCu-1677328905242)(null)]

图2:应用/网络模拟器工具中使用的模拟测试网络,也是由make check lab6运行的。 (有趣的事实:uun网络是David Mazieres的互联网切片,于1993年分配whois工具或链接的网站可以用来查询谁控制了每个IP地址的分配)。

Q & A

  • 我应该用什么数据结构来记录路由表?

    由你决定! 但不需要太过疯狂。每个数据报需要做O(n)个工作是完全可以接受的,其中n是路由表的条目数。如果你想做一些更有效的事情,我们鼓励你在优化之前先得到一个有效的实现,并仔细记录和评论你选择的任何实现。

  • 如何将以地址对象形式出现的IP地址转换为可以写入ARP消息的32位原始整数?

    使用Address::ipv4_numeric()方法。

  • 如何将一个以原始32位整数形式出现的IP地址转换为一个地址对象?

    使用 Address::from_ipv4_numeric()方法。

  • 如何将一个32位IP地址的最高n位(其中0≤n≤32) 与另一个32位IP地址的最重要的n位进行比较?

    这可能是这项任务中”最棘手”的部分,因为要让逻辑正确。也许值得在C++中写一个小的测试程序(一个简短的独立程序)或者在Sponge中添加一个测试,以验证你对相关的C++操作符的理解,并仔细检查你的逻辑。

    回顾一下,在C和C++中,将一个32位整数移位32位,可能会产生未定义行为。使用make_clean,然后在编译代码时打开sanitizer(cmake -DCMAKE_BUILD_TYPE=RelASan)以便在你提交之前尝试捕捉你的代码中任何未定义的行为。

    你可以通过在build目录中运行./apps/network simulator来直接运行路由器测试。

  • 如果路由器没有到目的地的路由,或者TTL为零,它是不是应该向数据报的源头发送一个ICMP错误信息?

    在现实生活中,是的,这将是有帮助的。但在这个实验里没有必要——丢弃数据报就足够了。(即使在现实生活中,也不是每个路由器都会在这些情况下向源头发送ICMP消息)。

  • 我如何运行本实验的测试套件?

    make check_lab6(两个测试)。或者你可以用make check运行整个测试套件(161个测试)。

  • 如果这个PDF出来后还有更多的FAQ,我在哪里可以看到?

    请定期查看网站(https://cs144.github.io/lab_faq.html)和Piazza。

有关CS144-Lab6的更多相关文章

  1. ruby-on-rails - Rails for Zombies Lab 4 > 练习 3 - 2

    我在第三个练习中停留在第四个RailsforZombies实验室。这是我的任务:创建将创建新僵尸的操作,然后重定向到创建的僵尸的显示页面。我有以下参数数组:params={:zombie=>{:name=>"Greg",:graveyard=>"TBA"}}我写了下面的代码作为解决方案:defcreate@zombie=Zombie.create@zombie.name=params[:zombie[:name]]@zombie.graveyard=params[:zombie[:graveyard]]@zombie.saveredirect_to(create_zombie_path

  2. c# - 从 global.asax.cs 调用 JavaScript 函数 - 2

    我想从global.asax.cs文件调用JavaScript函数(它在内部显示有关错误消息的弹出窗口)。这是我在Global.asax.cs文件中尝试的代码,protectedvoidApplication_Error(objectsender,EventArgse){System.Web.UI.Pagemypage=(System.Web.UI.Page)HttpContext.Current.Handler;mypage.RegisterStartupScript("alert","alert('test');");}但它不会调用警报,也不会在Firebug中给出任何错误或警告消

  3. javascript - 如何使 flash cs3、actionscript 将事件发送到 javascript? - 2

    我正在使用Flash在我的网站上播放.flv影片剪辑,但我希望在开始加载、开始播放和结束播放时让.swf发送在我的javascript中触发一个事件。在FlashCS3中使用Actionscript3.0执行此操作的最佳方法是什么? 最佳答案 您需要在HTML中使用“allowScriptAccess”flash变量。您可能希望使用“sameDomain”作为类型。请注意,如果您跨域,您还需要在服务器上托管一个名为“crossdomain.xml”的特殊文件,该文件启用此类脚本(Flash播放器将对此进行检查。更多信息,请访问htt

  4. javascript - 从 Javascript 获取 InDesign CS5 中的当前页码 - 2

    获取光标所在页面的DOM表达式是什么:app.activeDocument.currentPage? 最佳答案 你想要:app.activeWindow.activePage.name;这将为您提供字符串形式的实际页码名称(即,如果您的页面以罗马数字编号,那么这将为第五页提供“v”)。 关于javascript-从Javascript获取InDesignCS5中的当前页码,我们在StackOverflow上找到一个类似的问题: https://stackove

  5. VideoMAE:南大MCG&腾讯AI Lab 提出第一个视频版MAE框架,使用90%甚至95%遮挡,性能SOTA!... - 2

    关注公众号,发现CV技术之美本文分享论文『VideoMAE:MaskedAutoencodersareData-EfficientLearnersforSelf-SupervisedVideoPre-Training』,由南大王利民团队提出第一个VideoMAE框架,使用超高maskingratio(90%-95%),性能SOTA,代码已开源!详细信息如下:论文链接:https://arxiv.org/abs/2203.12602项目链接:https://github.com/MCG-NJU/VideoMAE    01     摘要为了在相对较小的数据集上实现卓越的性能,通常需要在超大规模数据

  6. MCDF实验4:魔龙的狂舞(从verilog到SV的入门lab4) - 2

    前言:验证结构与实验3是相同的,但需要验证的对象是完整的mcdf。对比之前新添加了reg寄存器模块(选择数据),formatter模块(数据打包)。种一棵树最好的时间是十年前,其次是现在。不是吗?实验3结构包含moinitor、checker、generator、initiator、test,这已经是一个完整的仿真结构,实验4可以说是实验3结构的复制粘贴。实验4将设计变得更复杂,添加了reg寄存器模块,formatter模块。验证过程完全相同,需要像实验3的验证过程一样对这两个模块也做仿真验证。设计中reg的功能是可以选择从哪个fifo接收数据,并且可以判断fifo余量(之前是margin),

  7. javascript - 在 Adob​​e InDesign CS5 中处理文本框中的文本 - 2

    如何在AdobeInDesignCS5中对文本框中的文本进行调整大小操作?主要目标是将每一行的最后一个单词分开,以实现自定义连字之类的功能。是否可以使用JavaScript(或ExtendScript?)?谢谢。更新我尝试在评论中描述我的问题我需要一个关于ExtendScript的示例,它会找到文本框尺寸并根据尺寸将文本连字符,然后我想在文本框调整大小时创建一个事件,这会在之后做同样的事情每次调整大小。 最佳答案 您可能可以编写脚本来执行您想要的操作,但该功能已内置于InDesign中。如果我对您的问题的理解正确,那是InDesig

  8. javascript - 在 Lab、Hcl 或任何感知均匀的颜色系统中改变色调和亮度 - 2

    在HSL颜色空间中更改色调、饱和度和亮度非常容易。参见thisjavascriptimplementationforexample.生成配色方案也很容易,例如:类似色、单色、三色、四色等。参见thisjavascriptimplementation.问题在于HSL颜色系统在感知上并不统一。实际上这是一个真正的问题。如果你不知道我在说什么,没关系。阅读:here和here.(这两篇文章真的很短,但很有值(value),你真的应该读一读)。解决方案是使用感知统一的色彩空间,例如:CieLab、CieLuv或Hcl。问题是这些颜色空间也有色域外的颜色(无法显示的颜色)。要在javascrip

  9. javascript - 如何使用 Adob​​e Indesign CS4 中的 Javascript 打开文本文件? - 2

    如何在InDesign中打开文本文件、读取内容,然后将内容插入到文档中? 最佳答案 这是从InDesign读取文件的示例。如果您还想写入文件,则需要以写入模式w来打开文件。//Choosethefilefromadialogvarfile=File.openDialog();//Oruseahardcodedpathtothefile//varfile=File("~/Desktop/helloworld.txt");//Openthefileforreadingfile.open("r");//Getthefirsttextfra

  10. xml - 无法生成临时类(结果=1)。错误 CS0030 : Cannot convert type - 2

    与thispost中的症状相同基本症状是,当我使用xsd.exe从我的XML创建我的C#类时,当我尝试反序列化它时,我得到这个错误(它编译,但在运行时产生错误)。Unabletogenerateatemporaryclass(result=1).ErrorCS0030:Cannotconverttype'ProductEnvironmentServerRolesServerRole[]'to'ProductEnvironmentServerRolesServerRole'原文(来自(xsd.exe)生成的.cs):[System.Xml.Serialization.XmlArrayAt

随机推荐