| // Copyright (c) 2012 The Chromium OS 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 "shill/modem.h" |
| |
| #include <vector> |
| |
| #include <base/stringprintf.h> |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| #include <mm/mm-modem.h> |
| #include <net/if.h> |
| #include <sys/ioctl.h> |
| |
| #include "shill/cellular.h" |
| #include "shill/cellular_capability_gsm.h" |
| #include "shill/dbus_property_matchers.h" |
| #include "shill/event_dispatcher.h" |
| #include "shill/manager.h" |
| #include "shill/mock_cellular.h" |
| #include "shill/mock_dbus_properties_proxy.h" |
| #include "shill/mock_device_info.h" |
| #include "shill/mock_modem.h" |
| #include "shill/mock_modem_info.h" |
| #include "shill/mock_rtnl_handler.h" |
| #include "shill/proxy_factory.h" |
| #include "shill/rtnl_handler.h" |
| |
| using std::string; |
| using std::vector; |
| using testing::_; |
| using testing::AnyNumber; |
| using testing::DoAll; |
| using testing::Return; |
| using testing::SetArgumentPointee; |
| using testing::StrEq; |
| using testing::StrictMock; |
| using testing::Test; |
| |
| namespace shill { |
| |
| namespace { |
| |
| const int kTestInterfaceIndex = 5; |
| const char kLinkName[] = "usb0"; |
| const char kOwner[] = ":1.18"; |
| const char kService[] = "org.chromium.ModemManager"; |
| const char kPath[] = "/org/chromium/ModemManager/Gobi/0"; |
| const unsigned char kAddress[] = {0x00, 0x01, 0x02, 0x03, 0x04, 0x05}; |
| const char kAddressAsString[] = "000102030405"; |
| |
| } // namespace |
| |
| class ModemTest : public Test { |
| public: |
| ModemTest() |
| : modem_info_(NULL, &dispatcher_, NULL, NULL, NULL), |
| device_info_(modem_info_.control_interface(), modem_info_.dispatcher(), |
| modem_info_.metrics(), modem_info_.manager()), |
| proxy_(new MockDBusPropertiesProxy()), |
| proxy_factory_(this), |
| modem_( |
| new StrictModem( |
| kOwner, |
| kService, |
| kPath, |
| &modem_info_)) {} |
| virtual void SetUp(); |
| virtual void TearDown(); |
| |
| void ReplaceSingletons() { |
| modem_->rtnl_handler_ = &rtnl_handler_; |
| } |
| |
| protected: |
| class TestProxyFactory : public ProxyFactory { |
| public: |
| explicit TestProxyFactory(ModemTest *test) : test_(test) {} |
| |
| virtual DBusPropertiesProxyInterface *CreateDBusPropertiesProxy( |
| const string &/*path*/, |
| const string &/*service*/) { |
| return test_->proxy_.release(); |
| } |
| |
| private: |
| ModemTest *test_; |
| }; |
| |
| EventDispatcher dispatcher_; |
| MockModemInfo modem_info_; |
| MockDeviceInfo device_info_; |
| scoped_ptr<MockDBusPropertiesProxy> proxy_; |
| TestProxyFactory proxy_factory_; |
| scoped_ptr<StrictModem> modem_; |
| MockRTNLHandler rtnl_handler_; |
| ByteString expected_address_; |
| }; |
| |
| void ModemTest::SetUp() { |
| EXPECT_EQ(kOwner, modem_->owner_); |
| EXPECT_EQ(kService, modem_->service_); |
| EXPECT_EQ(kPath, modem_->path_); |
| ReplaceSingletons(); |
| expected_address_ = ByteString(kAddress, arraysize(kAddress)); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(kLinkName)). |
| WillRepeatedly(Return(kTestInterfaceIndex)); |
| |
| EXPECT_CALL(*modem_info_.mock_manager(), device_info()) |
| .WillRepeatedly(Return(&device_info_)); |
| } |
| |
| void ModemTest::TearDown() { |
| modem_.reset(); |
| } |
| |
| TEST_F(ModemTest, PendingDevicePropertiesAndCreate) { |
| static const char kSentinel[] = "sentinel"; |
| static const uint32 kSentinelValue = 17; |
| |
| DBusInterfaceToProperties properties; |
| properties[MM_MODEM_INTERFACE][kSentinel].writer().append_uint32( |
| kSentinelValue); |
| |
| EXPECT_CALL(*modem_, GetLinkName(_, _)).WillRepeatedly(DoAll( |
| SetArgumentPointee<1>(string(kLinkName)), |
| Return(true))); |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(StrEq(kLinkName))). |
| WillRepeatedly(Return(kTestInterfaceIndex)); |
| |
| // The first time we call CreateDeviceFromModemProperties, |
| // GetMACAddress will fail. |
| EXPECT_CALL(device_info_, GetMACAddress(kTestInterfaceIndex, _)). |
| WillOnce(Return(false)); |
| EXPECT_CALL(*modem_, GetModemInterface()). |
| WillRepeatedly(Return(MM_MODEM_INTERFACE)); |
| modem_->CreateDeviceFromModemProperties(properties); |
| EXPECT_FALSE(modem_->device_.get()); |
| |
| // On the second time, we allow GetMACAddress to succeed. Now we |
| // expect a device to be built |
| EXPECT_CALL(device_info_, GetMACAddress(kTestInterfaceIndex, _)). |
| WillOnce(DoAll(SetArgumentPointee<1>(expected_address_), |
| Return(true))); |
| |
| // modem will take ownership |
| MockCellular *cellular = new MockCellular( |
| &modem_info_, |
| kLinkName, |
| kAddressAsString, |
| kTestInterfaceIndex, |
| Cellular::kTypeCDMA, |
| kOwner, |
| kService, |
| kPath, |
| ProxyFactory::GetInstance()); |
| |
| EXPECT_CALL(*modem_, |
| ConstructCellular(StrEq(kLinkName), |
| StrEq(kAddressAsString), |
| kTestInterfaceIndex)). |
| WillOnce(Return(cellular)); |
| |
| EXPECT_CALL(*cellular, OnDBusPropertiesChanged( |
| _, |
| HasDBusPropertyWithValueU32(kSentinel, kSentinelValue), |
| _)); |
| EXPECT_CALL(device_info_, RegisterDevice(_)); |
| modem_->OnDeviceInfoAvailable(kLinkName); |
| |
| EXPECT_TRUE(modem_->device_.get()); |
| |
| // Add expectations for the eventual |modem_| destruction. |
| EXPECT_CALL(*cellular, DestroyService()); |
| EXPECT_CALL(device_info_, DeregisterDevice(_)); |
| } |
| |
| TEST_F(ModemTest, EarlyDeviceProperties) { |
| // OnDeviceInfoAvailable called before |
| // CreateDeviceFromModemProperties: Do nothing |
| modem_->OnDeviceInfoAvailable(kLinkName); |
| EXPECT_FALSE(modem_->device_.get()); |
| } |
| |
| TEST_F(ModemTest, CreateDeviceEarlyFailures) { |
| DBusInterfaceToProperties properties; |
| |
| EXPECT_CALL(*modem_, ConstructCellular(_, _, _)).Times(0); |
| EXPECT_CALL(*modem_, GetModemInterface()). |
| WillRepeatedly(Return(MM_MODEM_INTERFACE)); |
| |
| // No modem interface properties: no device created |
| modem_->CreateDeviceFromModemProperties(properties); |
| EXPECT_FALSE(modem_->device_.get()); |
| |
| properties[MM_MODEM_INTERFACE] = DBusPropertiesMap(); |
| |
| // Link name, but no ifindex: no device created |
| EXPECT_CALL(*modem_, GetLinkName(_, _)).WillOnce(DoAll( |
| SetArgumentPointee<1>(string(kLinkName)), |
| Return(true))); |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(StrEq(kLinkName))).WillOnce( |
| Return(-1)); |
| modem_->CreateDeviceFromModemProperties(properties); |
| EXPECT_FALSE(modem_->device_.get()); |
| |
| // The params are good, but the device is blacklisted. |
| EXPECT_CALL(*modem_, GetLinkName(_, _)).WillOnce(DoAll( |
| SetArgumentPointee<1>(string(kLinkName)), |
| Return(true))); |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(StrEq(kLinkName))) |
| .WillOnce(Return(kTestInterfaceIndex)); |
| EXPECT_CALL(device_info_, GetMACAddress(kTestInterfaceIndex, _)) |
| .WillOnce(DoAll(SetArgumentPointee<1>(expected_address_), |
| Return(true))); |
| EXPECT_CALL(device_info_, IsDeviceBlackListed(kLinkName)) |
| .WillRepeatedly(Return(true)); |
| modem_->CreateDeviceFromModemProperties(properties); |
| EXPECT_FALSE(modem_->device_.get()); |
| |
| // No link name: see CreateDevicePPP. |
| } |
| |
| TEST_F(ModemTest, CreateDevicePPP) { |
| DBusInterfaceToProperties properties; |
| properties[MM_MODEM_INTERFACE] = DBusPropertiesMap(); |
| |
| string dev_name( |
| base::StringPrintf(Modem::kFakeDevNameFormat, Modem::fake_dev_serial_)); |
| |
| // |modem_| will take ownership. |
| MockCellular *cellular = new MockCellular( |
| &modem_info_, |
| dev_name, |
| Modem::kFakeDevAddress, |
| Modem::kFakeDevInterfaceIndex, |
| Cellular::kTypeUniversal, |
| kOwner, |
| kService, |
| kPath, |
| ProxyFactory::GetInstance()); |
| |
| EXPECT_CALL(*modem_, GetModemInterface()). |
| WillRepeatedly(Return(MM_MODEM_INTERFACE)); |
| // No link name: assumed to be a PPP dongle. |
| EXPECT_CALL(*modem_, GetLinkName(_, _)).WillOnce(Return(false)); |
| EXPECT_CALL(*modem_, |
| ConstructCellular(dev_name, |
| StrEq(Modem::kFakeDevAddress), |
| Modem::kFakeDevInterfaceIndex)). |
| WillOnce(Return(cellular)); |
| EXPECT_CALL(device_info_, RegisterDevice(_)); |
| |
| modem_->CreateDeviceFromModemProperties(properties); |
| EXPECT_TRUE(modem_->device_.get()); |
| |
| // Add expectations for the eventual |modem_| destruction. |
| EXPECT_CALL(*cellular, DestroyService()); |
| EXPECT_CALL(device_info_, DeregisterDevice(_)); |
| } |
| |
| TEST_F(ModemTest, GetDeviceParams) { |
| string mac_address; |
| int interface_index = 2; |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(_)).WillOnce(Return(-1)); |
| EXPECT_CALL(device_info_, GetMACAddress(_, _)).Times(AnyNumber()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_FALSE(modem_->GetDeviceParams(&mac_address, &interface_index)); |
| EXPECT_EQ(-1, interface_index); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(_)).WillOnce(Return(-2)); |
| EXPECT_CALL(device_info_, GetMACAddress(_, _)).Times(AnyNumber()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_FALSE(modem_->GetDeviceParams(&mac_address, &interface_index)); |
| EXPECT_EQ(-2, interface_index); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(_)).WillOnce(Return(1)); |
| EXPECT_CALL(device_info_, GetMACAddress(_, _)).WillOnce(Return(false)); |
| EXPECT_FALSE(modem_->GetDeviceParams(&mac_address, &interface_index)); |
| EXPECT_EQ(1, interface_index); |
| |
| EXPECT_CALL(rtnl_handler_, GetInterfaceIndex(_)).WillOnce(Return(2)); |
| EXPECT_CALL(device_info_, GetMACAddress(2, _)). |
| WillOnce(DoAll(SetArgumentPointee<1>(expected_address_), |
| Return(true))); |
| EXPECT_TRUE(modem_->GetDeviceParams(&mac_address, &interface_index)); |
| EXPECT_EQ(2, interface_index); |
| EXPECT_EQ(kAddressAsString, mac_address); |
| |
| } |
| |
| TEST_F(ModemTest, RejectPPPModem) { |
| // TODO(rochberg): Port this to ModemClassic |
| } |
| |
| } // namespace shill |