/* * Copyright (C) 2016, The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include #include #include #include #include "wificond/net/netlink_utils.h" #include "wificond/tests/mock_netlink_manager.h" using std::string; using std::unique_ptr; using std::vector; using testing::DoAll; using testing::NiceMock; using testing::Return; using testing::_; namespace android { namespace wificond { namespace { constexpr uint8_t kFakeMaxNumScanSSIDs = 10; constexpr uint8_t kFakeMaxNumSchedScanSSIDs = 16; constexpr uint8_t kFakeMaxMatchSets = 18; constexpr uint16_t kFakeFamilyId = 14; constexpr uint32_t kFakeFrequency1 = 2412; constexpr uint32_t kFakeFrequency2 = 2437; constexpr uint32_t kFakeFrequency3 = 2484; constexpr uint32_t kFakeFrequency4 = 5200; constexpr uint32_t kFakeFrequency5 = 5400; constexpr uint32_t kFakeFrequency6 = 5600; constexpr uint32_t kFakeSequenceNumber = 162; constexpr uint16_t kFakeWiphyIndex = 8; constexpr int kFakeErrorCode = EIO; const char kFakeInterfaceName[] = "testif0"; const uint32_t kFakeInterfaceIndex = 34; const uint32_t kFakeInterfaceIndex1 = 36; const uint8_t kFakeInterfaceMacAddress[] = {0x45, 0x54, 0xad, 0x67, 0x98, 0xf6}; const uint8_t kFakeInterfaceMacAddress1[] = {0x05, 0x04, 0xef, 0x27, 0x12, 0xff}; // Currently, control messages are only created by the kernel and sent to us. // Therefore NL80211Packet doesn't have corresponding constructor. // For test we manually create control messages using this helper function. NL80211Packet CreateControlMessageError(int error_code) { vector data; data.resize(NLMSG_HDRLEN + NLA_ALIGN(sizeof(int)), 0); // Initialize length field. nlmsghdr* nl_header = reinterpret_cast(data.data()); nl_header->nlmsg_len = data.size(); nl_header->nlmsg_type = NLMSG_ERROR; nl_header->nlmsg_seq = kFakeSequenceNumber; nl_header->nlmsg_pid = getpid(); int* error_field = reinterpret_cast(data.data() + NLMSG_HDRLEN); *error_field = -error_code; return NL80211Packet(data); } NL80211Packet CreateControlMessageAck() { return CreateControlMessageError(0); } } // namespace class NetlinkUtilsTest : public ::testing::Test { protected: std::unique_ptr> netlink_manager_; std::unique_ptr netlink_utils_; virtual void SetUp() { netlink_manager_.reset(new NiceMock()); netlink_utils_.reset(new NetlinkUtils(netlink_manager_.get())); ON_CALL(*netlink_manager_, GetSequenceNumber()).WillByDefault(Return(kFakeSequenceNumber)); ON_CALL(*netlink_manager_, GetFamilyId()).WillByDefault(Return(kFakeFamilyId)); } }; // This mocks the behavior of SendMessageAndGetResponses(), which returns a // vector of NL80211Packet using passed in pointer. ACTION_P(MakeupResponse, response) { // arg1 is the second parameter: vector>* responses. for (auto& pkt : response) { arg1->push_back(unique_ptr(new NL80211Packet(pkt))); } } TEST_F(NetlinkUtilsTest, CanGetWiphyIndex) { NL80211Packet new_wiphy( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_WIPHY, netlink_manager_->GetSequenceNumber(), getpid()); // Insert wiphy_index attribute. NL80211Attr wiphy_index_attr(NL80211_ATTR_WIPHY, kFakeWiphyIndex); new_wiphy.AddAttribute(wiphy_index_attr); // Mock a valid response from kernel. vector response = {new_wiphy}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); uint32_t wiphy_index; EXPECT_TRUE(netlink_utils_->GetWiphyIndex(&wiphy_index)); EXPECT_EQ(kFakeWiphyIndex, wiphy_index); } TEST_F(NetlinkUtilsTest, CanHandleGetWiphyIndexError) { // Mock an error response from kernel. vector response = {CreateControlMessageError(kFakeErrorCode)}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); uint32_t wiphy_index; EXPECT_FALSE(netlink_utils_->GetWiphyIndex(&wiphy_index)); } TEST_F(NetlinkUtilsTest, CanSetIntrerfaceMode) { // Mock a ACK response from kernel. vector response = {CreateControlMessageAck()}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); EXPECT_TRUE(netlink_utils_->SetInterfaceMode(kFakeInterfaceIndex, NetlinkUtils::STATION_MODE)); } TEST_F(NetlinkUtilsTest, CanHandleSetIntrerfaceModeError) { // Mock an error response from kernel. vector response = {CreateControlMessageError(kFakeErrorCode)}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); EXPECT_FALSE(netlink_utils_->SetInterfaceMode(kFakeInterfaceIndex, NetlinkUtils::STATION_MODE)); } TEST_F(NetlinkUtilsTest, CanGetInterfaces) { NL80211Packet new_interface( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_INTERFACE, netlink_manager_->GetSequenceNumber(), getpid()); // Insert interface name attribute. NL80211Attr if_name_attr(NL80211_ATTR_IFNAME, string(kFakeInterfaceName)); new_interface.AddAttribute(if_name_attr); // Insert interface index attribute. NL80211Attr if_index_attr(NL80211_ATTR_IFINDEX, kFakeInterfaceIndex); new_interface.AddAttribute(if_index_attr); // Insert mac address attribute. std::vector if_mac_addr; if_mac_addr.assign( kFakeInterfaceMacAddress, kFakeInterfaceMacAddress + sizeof(kFakeInterfaceMacAddress)); NL80211Attr> if_mac_attr(NL80211_ATTR_MAC, if_mac_addr); new_interface.AddAttribute(if_mac_attr); // Mock a valid response from kernel. vector response = {new_interface}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); vector interfaces; EXPECT_TRUE(netlink_utils_->GetInterfaces(kFakeWiphyIndex, &interfaces)); EXPECT_TRUE(interfaces.size() == 1); EXPECT_EQ(kFakeInterfaceIndex, interfaces[0].index); EXPECT_EQ(string(kFakeInterfaceName), interfaces[0].name); EXPECT_EQ(if_mac_addr, interfaces[0].mac_address); } TEST_F(NetlinkUtilsTest, SkipsPseudoDevicesWhenGetInterfaces) { // This might be a psuedo p2p interface without any interface index/name // attributes. NL80211Packet psuedo_interface( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_INTERFACE, netlink_manager_->GetSequenceNumber(), getpid()); psuedo_interface.AddAttribute(NL80211Attr( NL80211_ATTR_WDEV, 0)); // This is a regular client interface. NL80211Packet expected_interface( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_INTERFACE, netlink_manager_->GetSequenceNumber(), getpid()); expected_interface.AddAttribute(NL80211Attr( NL80211_ATTR_IFNAME, string(kFakeInterfaceName))); expected_interface.AddAttribute(NL80211Attr( NL80211_ATTR_IFINDEX, kFakeInterfaceIndex)); std::vector if_mac_addr; if_mac_addr.assign( kFakeInterfaceMacAddress, kFakeInterfaceMacAddress + sizeof(kFakeInterfaceMacAddress)); expected_interface.AddAttribute( NL80211Attr>(NL80211_ATTR_MAC, if_mac_addr)); // Kernel can send us the pseduo interface packet first vector response = {psuedo_interface, expected_interface}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); vector interfaces; EXPECT_TRUE(netlink_utils_->GetInterfaces(kFakeWiphyIndex, &interfaces)); EXPECT_TRUE(interfaces.size() == 1); EXPECT_EQ(kFakeInterfaceIndex, interfaces[0].index); EXPECT_EQ(string(kFakeInterfaceName), interfaces[0].name); EXPECT_EQ(if_mac_addr, interfaces[0].mac_address); } TEST_F(NetlinkUtilsTest, HandleP2p0WhenGetInterfaces) { NL80211Packet new_interface( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_INTERFACE, netlink_manager_->GetSequenceNumber(), getpid()); // Insert interface name attribute. NL80211Attr if_name_attr(NL80211_ATTR_IFNAME, string(kFakeInterfaceName)); new_interface.AddAttribute(if_name_attr); // Insert interface index attribute. new_interface.AddAttribute( NL80211Attr(NL80211_ATTR_IFINDEX, kFakeInterfaceIndex)); // Insert mac address attribute. std::vector if_mac_addr( kFakeInterfaceMacAddress, kFakeInterfaceMacAddress + sizeof(kFakeInterfaceMacAddress)); new_interface.AddAttribute( NL80211Attr>(NL80211_ATTR_MAC, if_mac_addr)); // Create a new interface packet for p2p0. NL80211Packet new_interface_p2p0( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_INTERFACE, netlink_manager_->GetSequenceNumber(), getpid()); // Insert interface name attribute. new_interface_p2p0.AddAttribute( NL80211Attr(NL80211_ATTR_IFNAME, "p2p0")); // Insert interface index attribute. new_interface_p2p0.AddAttribute( NL80211Attr(NL80211_ATTR_IFINDEX, kFakeInterfaceIndex1)); // Insert mac address attribute. std::vector if_mac_addr_p2p( kFakeInterfaceMacAddress1, kFakeInterfaceMacAddress1 + sizeof(kFakeInterfaceMacAddress1)); new_interface_p2p0.AddAttribute( NL80211Attr>(NL80211_ATTR_MAC, if_mac_addr_p2p)); // Mock response from kernel, including 2 interfaces. vector response = {new_interface_p2p0, new_interface}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); vector interfaces; EXPECT_TRUE(netlink_utils_->GetInterfaces(kFakeWiphyIndex, &interfaces)); EXPECT_TRUE(interfaces.size() == 2); EXPECT_EQ(kFakeInterfaceIndex1, interfaces[0].index); EXPECT_EQ(string("p2p0"), interfaces[0].name); EXPECT_EQ(if_mac_addr_p2p, interfaces[0].mac_address); EXPECT_EQ(kFakeInterfaceIndex, interfaces[1].index); EXPECT_EQ(string(kFakeInterfaceName), interfaces[1].name); EXPECT_EQ(if_mac_addr, interfaces[1].mac_address); } TEST_F(NetlinkUtilsTest, CanHandleGetInterfacesError) { // Mock an error response from kernel. vector response = {CreateControlMessageError(kFakeErrorCode)}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); vector interfaces; EXPECT_FALSE(netlink_utils_->GetInterfaces(kFakeWiphyIndex, &interfaces)); } TEST_F(NetlinkUtilsTest, CanGetWiphyInfo) { NL80211Packet new_wiphy( netlink_manager_->GetFamilyId(), NL80211_CMD_NEW_WIPHY, netlink_manager_->GetSequenceNumber(), getpid()); new_wiphy.AddAttribute(NL80211Attr(NL80211_ATTR_WIPHY, kFakeWiphyIndex)); // Insert band information to mock netlink response. NL80211NestedAttr freq_2g_1(1); NL80211NestedAttr freq_2g_2(2); NL80211NestedAttr freq_2g_3(3); NL80211NestedAttr freq_5g_1(4); NL80211NestedAttr freq_5g_2(5); NL80211NestedAttr freq_dfs_1(6); freq_2g_1.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency1)); freq_2g_2.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency2)); freq_2g_3.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency3)); freq_5g_1.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency4)); freq_5g_2.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency5)); // DFS frequency. freq_dfs_1.AddAttribute(NL80211Attr(NL80211_FREQUENCY_ATTR_FREQ, kFakeFrequency6)); freq_dfs_1.AddAttribute(NL80211Attr( NL80211_FREQUENCY_ATTR_DFS_STATE, NL80211_DFS_USABLE)); NL80211NestedAttr band_2g_freqs(NL80211_BAND_ATTR_FREQS); NL80211NestedAttr band_5g_freqs(NL80211_BAND_ATTR_FREQS); band_2g_freqs.AddAttribute(freq_2g_1); band_2g_freqs.AddAttribute(freq_2g_2); band_2g_freqs.AddAttribute(freq_2g_3); band_5g_freqs.AddAttribute(freq_5g_1); band_5g_freqs.AddAttribute(freq_5g_2); band_5g_freqs.AddAttribute(freq_dfs_1); NL80211NestedAttr band_2g_attr(1); NL80211NestedAttr band_5g_attr(2); band_2g_attr.AddAttribute(band_2g_freqs); band_5g_attr.AddAttribute(band_5g_freqs); NL80211NestedAttr band_attr(NL80211_ATTR_WIPHY_BANDS); band_attr.AddAttribute(band_2g_attr); band_attr.AddAttribute(band_5g_attr); new_wiphy.AddAttribute(band_attr); // Insert scan capabilities to mock netlink response. new_wiphy.AddAttribute(NL80211Attr(NL80211_ATTR_MAX_NUM_SCAN_SSIDS, kFakeMaxNumScanSSIDs)); new_wiphy.AddAttribute(NL80211Attr( NL80211_ATTR_MAX_NUM_SCHED_SCAN_SSIDS, kFakeMaxNumSchedScanSSIDs)); new_wiphy.AddAttribute(NL80211Attr(NL80211_ATTR_MAX_MATCH_SETS, kFakeMaxMatchSets)); // Insert wiphy features to mock netlink response. new_wiphy.AddAttribute(NL80211Attr( NL80211_ATTR_FEATURE_FLAGS, NL80211_FEATURE_SCAN_RANDOM_MAC_ADDR)); vector response = {new_wiphy}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); BandInfo band_info; ScanCapabilities scan_capabilities; WiphyFeatures wiphy_features; EXPECT_TRUE(netlink_utils_->GetWiphyInfo(kFakeWiphyIndex, &band_info, &scan_capabilities, &wiphy_features)); // Verify band information. vector band_2g_expected = {kFakeFrequency1, kFakeFrequency2, kFakeFrequency3}; vector band_5g_expected = {kFakeFrequency4, kFakeFrequency5}; vector band_dfs_expected = {kFakeFrequency6}; EXPECT_EQ(band_info.band_2g, band_2g_expected); EXPECT_EQ(band_info.band_5g, band_5g_expected); EXPECT_EQ(band_info.band_dfs, band_dfs_expected); // Verify scan capabilities. EXPECT_EQ(scan_capabilities.max_num_scan_ssids, kFakeMaxNumScanSSIDs); EXPECT_EQ(scan_capabilities.max_num_sched_scan_ssids, kFakeMaxNumSchedScanSSIDs); EXPECT_EQ(scan_capabilities.max_match_sets, kFakeMaxMatchSets); // Verify wiphy features. EXPECT_TRUE(wiphy_features.supports_random_mac_oneshot_scan); EXPECT_FALSE(wiphy_features.supports_random_mac_sched_scan); } TEST_F(NetlinkUtilsTest, CanHandleGetWiphyInfoError) { // Mock an error response from kernel. vector response = {CreateControlMessageError(kFakeErrorCode)}; EXPECT_CALL(*netlink_manager_, SendMessageAndGetResponses(_, _)). WillOnce(DoAll(MakeupResponse(response), Return(true))); BandInfo band_info; ScanCapabilities scan_capabilities; WiphyFeatures wiphy_features; EXPECT_FALSE(netlink_utils_->GetWiphyInfo(kFakeWiphyIndex, &band_info, &scan_capabilities, &wiphy_features)); } } // namespace wificond } // namespace android