| // Copyright (c) 2019 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "quic/qbone/platform/rtnetlink_message.h" | 
 |  | 
 | #include <net/if_arp.h> | 
 |  | 
 | #include "quic/platform/api/quic_ip_address.h" | 
 | #include "quic/platform/api/quic_test.h" | 
 |  | 
 | namespace quic { | 
 | namespace { | 
 |  | 
 | using ::testing::StrEq; | 
 |  | 
 | TEST(RtnetlinkMessageTest, LinkMessageCanBeCreatedForGetOperation) { | 
 |   uint16_t flags = NLM_F_REQUEST | NLM_F_ROOT | NLM_F_MATCH; | 
 |   uint32_t seq = 42; | 
 |   uint32_t pid = 7; | 
 |   auto message = LinkMessage::New(RtnetlinkMessage::Operation::GET, flags, seq, | 
 |                                   pid, nullptr); | 
 |  | 
 |   // No rtattr appended. | 
 |   EXPECT_EQ(1, message.IoVecSize()); | 
 |  | 
 |   // nlmsghdr is built properly. | 
 |   auto iov = message.BuildIoVec(); | 
 |   EXPECT_EQ(NLMSG_SPACE(sizeof(struct rtgenmsg)), iov[0].iov_len); | 
 |   auto* netlink_message = reinterpret_cast<struct nlmsghdr*>(iov[0].iov_base); | 
 |   EXPECT_EQ(NLMSG_LENGTH(sizeof(struct rtgenmsg)), netlink_message->nlmsg_len); | 
 |   EXPECT_EQ(RTM_GETLINK, netlink_message->nlmsg_type); | 
 |   EXPECT_EQ(flags, netlink_message->nlmsg_flags); | 
 |   EXPECT_EQ(seq, netlink_message->nlmsg_seq); | 
 |   EXPECT_EQ(pid, netlink_message->nlmsg_pid); | 
 |  | 
 |   // We actually included rtgenmsg instead of the passed in ifinfomsg since this | 
 |   // is a GET operation. | 
 |   EXPECT_EQ(NLMSG_LENGTH(sizeof(struct rtgenmsg)), netlink_message->nlmsg_len); | 
 | } | 
 |  | 
 | TEST(RtnetlinkMessageTest, LinkMessageCanBeCreatedForNewOperation) { | 
 |   struct ifinfomsg interface_info_header = {AF_INET, /* pad */ 0, ARPHRD_TUNNEL, | 
 |                                             3,       0,           0xffffffff}; | 
 |   uint16_t flags = NLM_F_REQUEST | NLM_F_ROOT | NLM_F_MATCH; | 
 |   uint32_t seq = 42; | 
 |   uint32_t pid = 7; | 
 |   auto message = LinkMessage::New(RtnetlinkMessage::Operation::NEW, flags, seq, | 
 |                                   pid, &interface_info_header); | 
 |  | 
 |   std::string device_name = "device0"; | 
 |   message.AppendAttribute(IFLA_IFNAME, device_name.c_str(), device_name.size()); | 
 |  | 
 |   // One rtattr appended. | 
 |   EXPECT_EQ(2, message.IoVecSize()); | 
 |  | 
 |   // nlmsghdr is built properly. | 
 |   auto iov = message.BuildIoVec(); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct ifinfomsg))), | 
 |             iov[0].iov_len); | 
 |   auto* netlink_message = reinterpret_cast<struct nlmsghdr*>(iov[0].iov_base); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct ifinfomsg))) + | 
 |                 RTA_LENGTH(device_name.size()), | 
 |             netlink_message->nlmsg_len); | 
 |   EXPECT_EQ(RTM_NEWLINK, netlink_message->nlmsg_type); | 
 |   EXPECT_EQ(flags, netlink_message->nlmsg_flags); | 
 |   EXPECT_EQ(seq, netlink_message->nlmsg_seq); | 
 |   EXPECT_EQ(pid, netlink_message->nlmsg_pid); | 
 |  | 
 |   // ifinfomsg is included properly. | 
 |   auto* parsed_header = | 
 |       reinterpret_cast<struct ifinfomsg*>(NLMSG_DATA(netlink_message)); | 
 |   EXPECT_EQ(interface_info_header.ifi_family, parsed_header->ifi_family); | 
 |   EXPECT_EQ(interface_info_header.ifi_type, parsed_header->ifi_type); | 
 |   EXPECT_EQ(interface_info_header.ifi_index, parsed_header->ifi_index); | 
 |   EXPECT_EQ(interface_info_header.ifi_flags, parsed_header->ifi_flags); | 
 |   EXPECT_EQ(interface_info_header.ifi_change, parsed_header->ifi_change); | 
 |  | 
 |   // rtattr is handled properly. | 
 |   EXPECT_EQ(RTA_SPACE(device_name.size()), iov[1].iov_len); | 
 |   auto* rta = reinterpret_cast<struct rtattr*>(iov[1].iov_base); | 
 |   EXPECT_EQ(IFLA_IFNAME, rta->rta_type); | 
 |   EXPECT_EQ(RTA_LENGTH(device_name.size()), rta->rta_len); | 
 |   EXPECT_THAT(device_name, | 
 |               StrEq(std::string(reinterpret_cast<char*>(RTA_DATA(rta)), | 
 |                                 RTA_PAYLOAD(rta)))); | 
 | } | 
 |  | 
 | TEST(RtnetlinkMessageTest, AddressMessageCanBeCreatedForGetOperation) { | 
 |   uint16_t flags = NLM_F_REQUEST | NLM_F_ROOT | NLM_F_MATCH; | 
 |   uint32_t seq = 42; | 
 |   uint32_t pid = 7; | 
 |   auto message = AddressMessage::New(RtnetlinkMessage::Operation::GET, flags, | 
 |                                      seq, pid, nullptr); | 
 |  | 
 |   // No rtattr appended. | 
 |   EXPECT_EQ(1, message.IoVecSize()); | 
 |  | 
 |   // nlmsghdr is built properly. | 
 |   auto iov = message.BuildIoVec(); | 
 |   EXPECT_EQ(NLMSG_SPACE(sizeof(struct rtgenmsg)), iov[0].iov_len); | 
 |   auto* netlink_message = reinterpret_cast<struct nlmsghdr*>(iov[0].iov_base); | 
 |   EXPECT_EQ(NLMSG_LENGTH(sizeof(struct rtgenmsg)), netlink_message->nlmsg_len); | 
 |   EXPECT_EQ(RTM_GETADDR, netlink_message->nlmsg_type); | 
 |   EXPECT_EQ(flags, netlink_message->nlmsg_flags); | 
 |   EXPECT_EQ(seq, netlink_message->nlmsg_seq); | 
 |   EXPECT_EQ(pid, netlink_message->nlmsg_pid); | 
 |  | 
 |   // We actually included rtgenmsg instead of the passed in ifinfomsg since this | 
 |   // is a GET operation. | 
 |   EXPECT_EQ(NLMSG_LENGTH(sizeof(struct rtgenmsg)), netlink_message->nlmsg_len); | 
 | } | 
 |  | 
 | TEST(RtnetlinkMessageTest, AddressMessageCanBeCreatedForNewOperation) { | 
 |   struct ifaddrmsg interface_address_header = {AF_INET, | 
 |                                                /* prefixlen */ 24, | 
 |                                                /* flags */ 0, | 
 |                                                /* scope */ RT_SCOPE_LINK, | 
 |                                                /* index */ 4}; | 
 |   uint16_t flags = NLM_F_REQUEST | NLM_F_ROOT | NLM_F_MATCH; | 
 |   uint32_t seq = 42; | 
 |   uint32_t pid = 7; | 
 |   auto message = AddressMessage::New(RtnetlinkMessage::Operation::NEW, flags, | 
 |                                      seq, pid, &interface_address_header); | 
 |  | 
 |   QuicIpAddress ip; | 
 |   QUICHE_CHECK(ip.FromString("10.0.100.3")); | 
 |   message.AppendAttribute(IFA_ADDRESS, ip.ToPackedString().c_str(), | 
 |                           ip.ToPackedString().size()); | 
 |  | 
 |   // One rtattr is appended. | 
 |   EXPECT_EQ(2, message.IoVecSize()); | 
 |  | 
 |   // nlmsghdr is built properly. | 
 |   auto iov = message.BuildIoVec(); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct ifaddrmsg))), | 
 |             iov[0].iov_len); | 
 |   auto* netlink_message = reinterpret_cast<struct nlmsghdr*>(iov[0].iov_base); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct ifaddrmsg))) + | 
 |                 RTA_LENGTH(ip.ToPackedString().size()), | 
 |             netlink_message->nlmsg_len); | 
 |   EXPECT_EQ(RTM_NEWADDR, netlink_message->nlmsg_type); | 
 |   EXPECT_EQ(flags, netlink_message->nlmsg_flags); | 
 |   EXPECT_EQ(seq, netlink_message->nlmsg_seq); | 
 |   EXPECT_EQ(pid, netlink_message->nlmsg_pid); | 
 |  | 
 |   // ifaddrmsg is included properly. | 
 |   auto* parsed_header = | 
 |       reinterpret_cast<struct ifaddrmsg*>(NLMSG_DATA(netlink_message)); | 
 |   EXPECT_EQ(interface_address_header.ifa_family, parsed_header->ifa_family); | 
 |   EXPECT_EQ(interface_address_header.ifa_prefixlen, | 
 |             parsed_header->ifa_prefixlen); | 
 |   EXPECT_EQ(interface_address_header.ifa_flags, parsed_header->ifa_flags); | 
 |   EXPECT_EQ(interface_address_header.ifa_scope, parsed_header->ifa_scope); | 
 |   EXPECT_EQ(interface_address_header.ifa_index, parsed_header->ifa_index); | 
 |  | 
 |   // rtattr is handled properly. | 
 |   EXPECT_EQ(RTA_SPACE(ip.ToPackedString().size()), iov[1].iov_len); | 
 |   auto* rta = reinterpret_cast<struct rtattr*>(iov[1].iov_base); | 
 |   EXPECT_EQ(IFA_ADDRESS, rta->rta_type); | 
 |   EXPECT_EQ(RTA_LENGTH(ip.ToPackedString().size()), rta->rta_len); | 
 |   EXPECT_THAT(ip.ToPackedString(), | 
 |               StrEq(std::string(reinterpret_cast<char*>(RTA_DATA(rta)), | 
 |                                 RTA_PAYLOAD(rta)))); | 
 | } | 
 |  | 
 | TEST(RtnetlinkMessageTest, RouteMessageCanBeCreatedFromNewOperation) { | 
 |   struct rtmsg route_message_header = {AF_INET6, | 
 |                                        /* rtm_dst_len */ 48, | 
 |                                        /* rtm_src_len */ 0, | 
 |                                        /* rtm_tos */ 0, | 
 |                                        /* rtm_table */ RT_TABLE_MAIN, | 
 |                                        /* rtm_protocol */ RTPROT_STATIC, | 
 |                                        /* rtm_scope */ RT_SCOPE_LINK, | 
 |                                        /* rtm_type */ RTN_LOCAL, | 
 |                                        /* rtm_flags */ 0}; | 
 |   uint16_t flags = NLM_F_REQUEST | NLM_F_ROOT | NLM_F_MATCH; | 
 |   uint32_t seq = 42; | 
 |   uint32_t pid = 7; | 
 |   auto message = RouteMessage::New(RtnetlinkMessage::Operation::NEW, flags, seq, | 
 |                                    pid, &route_message_header); | 
 |  | 
 |   QuicIpAddress preferred_source; | 
 |   QUICHE_CHECK(preferred_source.FromString("ff80::1")); | 
 |   message.AppendAttribute(RTA_PREFSRC, | 
 |                           preferred_source.ToPackedString().c_str(), | 
 |                           preferred_source.ToPackedString().size()); | 
 |  | 
 |   // One rtattr is appended. | 
 |   EXPECT_EQ(2, message.IoVecSize()); | 
 |  | 
 |   // nlmsghdr is built properly | 
 |   auto iov = message.BuildIoVec(); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct rtmsg))), iov[0].iov_len); | 
 |   auto* netlink_message = reinterpret_cast<struct nlmsghdr*>(iov[0].iov_base); | 
 |   EXPECT_EQ(NLMSG_ALIGN(NLMSG_LENGTH(sizeof(struct rtmsg))) + | 
 |                 RTA_LENGTH(preferred_source.ToPackedString().size()), | 
 |             netlink_message->nlmsg_len); | 
 |   EXPECT_EQ(RTM_NEWROUTE, netlink_message->nlmsg_type); | 
 |   EXPECT_EQ(flags, netlink_message->nlmsg_flags); | 
 |   EXPECT_EQ(seq, netlink_message->nlmsg_seq); | 
 |   EXPECT_EQ(pid, netlink_message->nlmsg_pid); | 
 |  | 
 |   // rtmsg is included properly. | 
 |   auto* parsed_header = | 
 |       reinterpret_cast<struct rtmsg*>(NLMSG_DATA(netlink_message)); | 
 |   EXPECT_EQ(route_message_header.rtm_family, parsed_header->rtm_family); | 
 |   EXPECT_EQ(route_message_header.rtm_dst_len, parsed_header->rtm_dst_len); | 
 |   EXPECT_EQ(route_message_header.rtm_src_len, parsed_header->rtm_src_len); | 
 |   EXPECT_EQ(route_message_header.rtm_tos, parsed_header->rtm_tos); | 
 |   EXPECT_EQ(route_message_header.rtm_table, parsed_header->rtm_table); | 
 |   EXPECT_EQ(route_message_header.rtm_protocol, parsed_header->rtm_protocol); | 
 |   EXPECT_EQ(route_message_header.rtm_scope, parsed_header->rtm_scope); | 
 |   EXPECT_EQ(route_message_header.rtm_type, parsed_header->rtm_type); | 
 |   EXPECT_EQ(route_message_header.rtm_flags, parsed_header->rtm_flags); | 
 |  | 
 |   // rtattr is handled properly. | 
 |   EXPECT_EQ(RTA_SPACE(preferred_source.ToPackedString().size()), | 
 |             iov[1].iov_len); | 
 |   auto* rta = reinterpret_cast<struct rtattr*>(iov[1].iov_base); | 
 |   EXPECT_EQ(RTA_PREFSRC, rta->rta_type); | 
 |   EXPECT_EQ(RTA_LENGTH(preferred_source.ToPackedString().size()), rta->rta_len); | 
 |   EXPECT_THAT(preferred_source.ToPackedString(), | 
 |               StrEq(std::string(reinterpret_cast<char*>(RTA_DATA(rta)), | 
 |                                 RTA_PAYLOAD(rta)))); | 
 | } | 
 |  | 
 | }  // namespace | 
 | }  // namespace quic |