//
// Copyright (C) 2015 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 <dhcp_client/device_info.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <shill/net/byte_string.h>
#include <shill/net/mock_sockets.h>
#include <shill/net/mock_rtnl_handler.h>
using shill::ByteString;
using shill::MockRTNLHandler;
using shill::MockSockets;
using ::testing::_;
using ::testing::DoAll;
using ::testing::ElementsAreArray;
using ::testing::Return;
namespace {
const int kFakeFd = 99;
const unsigned int kFakeInterfaceIndex = 1;
const std::string kFakeDeviceName = "eth0";
const std::string kFakeLongDeviceName = "a_long_device_name";
const uint8_t kFakeMacAddress[] = {0x00, 0x01, 0x02, 0xaa, 0xbb, 0xcc};
}
namespace dhcp_client {
class DeviceInfoTest : public testing::Test {
public:
DeviceInfoTest() {}
void SetUp() {
device_info_ = DeviceInfo::GetInstance();
sockets_ = new MockSockets();
device_info_->sockets_.reset(sockets_);
device_info_->rtnl_handler_ = &rtnl_handler_;
}
protected:
DeviceInfo* device_info_;
MockRTNLHandler rtnl_handler_;
MockSockets* sockets_; // Owned by device_info_.
};
ACTION_P(SetIfreq, ifr) {
struct ifreq* const ifr_arg = static_cast<struct ifreq*>(arg2);
*ifr_arg = ifr;
}
MATCHER_P(IfreqEquals, ifname, "") {
const struct ifreq* const ifr = static_cast<struct ifreq*>(arg);
return (ifr != nullptr) &&
(strcmp(ifname, ifr->ifr_name) == 0);
}
TEST_F(DeviceInfoTest, GetDeviceInfoSucceed) {
ByteString mac_address;
unsigned int interface_index;
struct ifreq ifr;
memcpy(ifr.ifr_hwaddr.sa_data, kFakeMacAddress, sizeof(kFakeMacAddress));
EXPECT_CALL(*sockets_, Socket(AF_INET, SOCK_DGRAM, 0))
.WillOnce(Return(kFakeFd));
EXPECT_CALL(*sockets_, Ioctl(kFakeFd,
SIOCGIFHWADDR,
IfreqEquals(kFakeDeviceName.c_str())))
.WillOnce(DoAll(SetIfreq(ifr), Return(0)));
EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(kFakeDeviceName))
.WillOnce(Return(kFakeInterfaceIndex));
EXPECT_TRUE(device_info_->GetDeviceInfo(kFakeDeviceName,
&mac_address,
&interface_index));
EXPECT_EQ(interface_index, kFakeInterfaceIndex);
EXPECT_THAT(kFakeMacAddress,
ElementsAreArray(mac_address.GetConstData(),
sizeof(kFakeMacAddress)));
}
TEST_F(DeviceInfoTest, GetDeviceInfoNameTooLong) {
ByteString mac_address;
unsigned int interface_index;
EXPECT_FALSE(device_info_->GetDeviceInfo(kFakeLongDeviceName,
&mac_address,
&interface_index));
}
TEST_F(DeviceInfoTest, GetDeviceInfoFailedToCreateSocket) {
ByteString mac_address;
unsigned int interface_index;
EXPECT_CALL(*sockets_, Socket(AF_INET, SOCK_DGRAM, 0)).WillOnce(Return(-1));
EXPECT_FALSE(device_info_->GetDeviceInfo(kFakeDeviceName,
&mac_address,
&interface_index));
}
TEST_F(DeviceInfoTest, GetDeviceInfoFailedToGetHardwareAddr) {
ByteString mac_address;
unsigned int interface_index;
EXPECT_CALL(*sockets_, Socket(AF_INET, SOCK_DGRAM, 0))
.WillOnce(Return(kFakeFd));
EXPECT_CALL(*sockets_, Ioctl(kFakeFd, SIOCGIFHWADDR, _)).WillOnce(Return(-1));
EXPECT_FALSE(device_info_->GetDeviceInfo(kFakeDeviceName,
&mac_address,
&interface_index));
}
TEST_F(DeviceInfoTest, GetDeviceInfoFailedToGetInterfaceIndex) {
ByteString mac_address;
unsigned int interface_index;
EXPECT_CALL(*sockets_, Socket(AF_INET, SOCK_DGRAM, 0))
.WillOnce(Return(kFakeFd));
EXPECT_CALL(*sockets_, Ioctl(kFakeFd, SIOCGIFHWADDR, _)).WillOnce(Return(0));
EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(kFakeDeviceName))
.WillOnce(Return(-1));
EXPECT_FALSE(device_info_->GetDeviceInfo(kFakeDeviceName,
&mac_address,
&interface_index));
}
} // namespace dhcp_client