| // Copyright 2018 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/cellular/cellular_capability_3gpp.h" |
| |
| #include <map> |
| #include <memory> |
| #include <string> |
| #include <tuple> |
| #include <utility> |
| #include <vector> |
| |
| #include <base/bind.h> |
| #include <base/check.h> |
| #include <base/strings/string_util.h> |
| #include <base/strings/stringprintf.h> |
| #include <chromeos/dbus/service_constants.h> |
| #include <ModemManager/ModemManager.h> |
| |
| #include "shill/cellular/cellular.h" |
| #include "shill/cellular/cellular_bearer.h" |
| #include "shill/cellular/cellular_service.h" |
| #include "shill/cellular/cellular_service_provider.h" |
| #include "shill/cellular/mock_cellular.h" |
| #include "shill/cellular/mock_cellular_service.h" |
| #include "shill/cellular/mock_mm1_modem_location_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_modem3gpp_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_signal_proxy.h" |
| #include "shill/cellular/mock_mm1_modem_simple_proxy.h" |
| #include "shill/cellular/mock_mm1_sim_proxy.h" |
| #include "shill/cellular/mock_mobile_operator_info.h" |
| #include "shill/cellular/mock_modem_info.h" |
| #include "shill/cellular/mock_pending_activation_store.h" |
| #include "shill/dbus/dbus_properties_proxy.h" |
| #include "shill/dbus/fake_properties_proxy.h" |
| #include "shill/error.h" |
| #include "shill/fake_store.h" |
| #include "shill/mock_adaptors.h" |
| #include "shill/mock_control.h" |
| #include "shill/mock_device_info.h" |
| #include "shill/mock_event_dispatcher.h" |
| #include "shill/mock_manager.h" |
| #include "shill/mock_metrics.h" |
| #include "shill/mock_profile.h" |
| #include "shill/test_event_dispatcher.h" |
| #include "shill/testing.h" |
| |
| using testing::_; |
| using testing::AnyNumber; |
| using testing::InSequence; |
| using testing::Invoke; |
| using testing::Mock; |
| using testing::NiceMock; |
| using testing::Return; |
| using testing::ReturnRef; |
| using testing::SaveArg; |
| |
| namespace shill { |
| |
| namespace { |
| |
| const uint32_t kAccessTechnologies = |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE | MM_MODEM_ACCESS_TECHNOLOGY_HSPA_PLUS; |
| const char kActiveBearerPathPrefix[] = "/bearer/active"; |
| constexpr char kDeviceId[] = "<device_id>"; |
| const char kEid[] = "310100000002"; |
| const char kIccid[] = "1234567890"; |
| const char kImei[] = "999911110000"; |
| const char kImsi[] = "310100000001"; |
| const char kInactiveBearerPathPrefix[] = "/bearer/inactive"; |
| const char kSimPathPrefix[] = "/foo/sim"; |
| const RpcIdentifier kSimPath1("/foo/sim/1"); |
| const RpcIdentifier kSimPath2("/foo/sim/2"); |
| |
| } // namespace |
| |
| MATCHER_P(HasApn, expected_apn, "") { |
| return arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectApn) && |
| expected_apn == |
| arg.template Get<std::string>(CellularCapability3gpp::kConnectApn); |
| } |
| |
| MATCHER(HasNoUser, "") { |
| return !arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectUser); |
| } |
| |
| MATCHER_P(HasUser, expected_user, "") { |
| return arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectUser) && |
| expected_user == arg.template Get<std::string>( |
| CellularCapability3gpp::kConnectUser); |
| } |
| |
| MATCHER(HasNoPassword, "") { |
| return !arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectPassword); |
| } |
| |
| MATCHER_P(HasPassword, expected_password, "") { |
| return arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectPassword) && |
| expected_password == arg.template Get<std::string>( |
| CellularCapability3gpp::kConnectPassword); |
| } |
| |
| MATCHER(HasNoAllowedAuth, "") { |
| return !arg.template Contains<std::string>( |
| CellularCapability3gpp::kConnectAllowedAuth); |
| } |
| |
| MATCHER_P(HasAllowedAuth, expected_authentication, "") { |
| return arg.template Contains<uint32_t>( |
| CellularCapability3gpp::kConnectAllowedAuth) && |
| expected_authentication == |
| arg.template Get<uint32_t>( |
| CellularCapability3gpp::kConnectAllowedAuth); |
| } |
| |
| MATCHER(HasNoIpType, "") { |
| return !arg.template Contains<uint32_t>( |
| CellularCapability3gpp::kConnectIpType); |
| } |
| |
| MATCHER_P(HasIpType, expected_ip_type, "") { |
| return arg.template Contains<uint32_t>( |
| CellularCapability3gpp::kConnectIpType) && |
| expected_ip_type == |
| arg.template Get<uint32_t>(CellularCapability3gpp::kConnectIpType); |
| } |
| |
| class CellularCapability3gppTest : public testing::TestWithParam<std::string> { |
| public: |
| CellularCapability3gppTest() |
| : control_interface_(this), |
| manager_(&control_interface_, &dispatcher_, &metrics_), |
| device_info_(&manager_), |
| modem_info_(&control_interface_, &manager_), |
| modem_3gpp_proxy_(new NiceMock<mm1::MockModemModem3gppProxy>()), |
| modem_proxy_(new mm1::MockModemProxy()), |
| modem_signal_proxy_(new NiceMock<mm1::MockModemSignalProxy>()), |
| modem_simple_proxy_(new NiceMock<mm1::MockModemSimpleProxy>()), |
| profile_(new NiceMock<MockProfile>(&manager_)), |
| mock_home_provider_info_(nullptr), |
| mock_serving_operator_info_(nullptr) { |
| cellular_service_provider_.set_profile_for_testing(profile_); |
| } |
| |
| ~CellularCapability3gppTest() override = default; |
| |
| void SetUp() override { |
| EXPECT_CALL(*modem_proxy_, set_state_changed_callback(_)) |
| .Times(AnyNumber()); |
| EXPECT_CALL(manager_, device_info()).WillRepeatedly(Return(&device_info_)); |
| EXPECT_CALL(manager_, modem_info()).WillRepeatedly(Return(&modem_info_)); |
| |
| cellular_ = new Cellular(&manager_, "", "00:01:02:03:04:05", 0, |
| Cellular::kType3gpp, "", RpcIdentifier("")); |
| service_ = new MockCellularService(&manager_, cellular_); |
| device_adaptor_ = static_cast<DeviceMockAdaptor*>(cellular_->adaptor()); |
| capability_ = static_cast<CellularCapability3gpp*>( |
| cellular_->capability_for_testing()); |
| cellular_->SetServiceForTesting(service_); |
| cellular_service_provider_.Start(); |
| metrics_.RegisterDevice(cellular_->interface_index(), |
| Technology::kCellular); |
| |
| EXPECT_CALL(*service_, activation_state()) |
| .WillRepeatedly(ReturnRef(kActivationStateUnknown)); |
| EXPECT_CALL(*service_, SetStrength(0)).Times(AnyNumber()); |
| |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_CALL(manager_, cellular_service_provider()) |
| .WillRepeatedly(Return(&cellular_service_provider_)); |
| |
| EXPECT_CALL(*profile_, GetConstStorage()) |
| .WillRepeatedly(Return(&profile_storage_)); |
| EXPECT_CALL(*profile_, GetStorage()) |
| .WillRepeatedly(Return(&profile_storage_)); |
| |
| SetMockMobileOperatorInfoObjects(); |
| } |
| |
| void TearDown() override { |
| metrics_.DeregisterDevice(cellular_->interface_index()); |
| cellular_service_provider_.Stop(); |
| metrics_.RegisterDevice(cellular_->interface_index(), |
| Technology::kCellular); |
| cellular_->SetServiceForTesting(nullptr); |
| service_ = nullptr; |
| CHECK(cellular_->HasOneRef()); |
| cellular_ = nullptr; |
| device_adaptor_ = nullptr; |
| } |
| |
| void VerifyAndSetActivationExpectations() { |
| Mock::VerifyAndClearExpectations( |
| modem_info_.mock_pending_activation_store()); |
| |
| // kStateUnknown leads to minimal extra work in maintaining |
| // activation state. |
| ON_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .WillByDefault(Return(PendingActivationStore::kStateUnknown)); |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .Times(AnyNumber()); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivated)) |
| .Times(AnyNumber()); |
| } |
| |
| // Saves |sim_properties| for |path| to be provided by FakePropertiesProxy. |
| void SetSimProperties(const RpcIdentifier& path, |
| const KeyValueStore& sim_properties) { |
| sim_paths_.push_back(path); |
| sim_properties_.push_back(sim_properties); |
| } |
| |
| // Calls capability_->OnPropertiesChanged with Modem.SIM = |path|. |
| void SetSimPath(const RpcIdentifier& path) { |
| KeyValueStore modem_properties; |
| modem_properties.Set<RpcIdentifier>(MM_MODEM_PROPERTY_SIM, path); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| // Calls capability_->OnPropertiesChanged with Modem.SIM = |path| and |
| // Modem.SIMSLOTS = |sim_properties_|. |
| void UpdateSims(const RpcIdentifier& path) { |
| KeyValueStore modem_properties; |
| modem_properties.Set<RpcIdentifier>(MM_MODEM_PROPERTY_SIM, path); |
| RpcIdentifiers slots; |
| for (const auto& path : sim_paths_) |
| slots.push_back(path); |
| modem_properties.Set<RpcIdentifiers>(MM_MODEM_PROPERTY_SIMSLOTS, slots); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| // Sets up a single SIM path and properties. |
| void SetSimPropertiesAndPath(const RpcIdentifier& path, |
| const KeyValueStore& sim_properties) { |
| SetSimProperties(path, sim_properties); |
| UpdateSims(path); |
| } |
| |
| void SetCellularSimProperties(const Cellular::SimProperties& sim_properties) { |
| std::vector<Cellular::SimProperties> slot_properties; |
| slot_properties.push_back(sim_properties); |
| cellular_->SetSimProperties(slot_properties, 0u); |
| } |
| |
| void SetDefaultCellularSimProperties() { |
| Cellular::SimProperties sim_properties; |
| sim_properties.eid = kEid; |
| sim_properties.iccid = kIccid; |
| sim_properties.imsi = kImsi; |
| SetCellularSimProperties(sim_properties); |
| } |
| |
| void ClearCellularSimProperties() { |
| SetCellularSimProperties(Cellular::SimProperties()); |
| } |
| |
| void ClearCapabilitySimProperties() { |
| sim_paths_.clear(); |
| sim_properties_.clear(); |
| UpdateSims(RpcIdentifier()); |
| } |
| |
| void CreateService() { |
| // The following constants are never directly accessed by the tests. |
| const char kFriendlyServiceName[] = "default_test_service_name"; |
| const char kOperatorCode[] = "10010"; |
| const char kOperatorName[] = "default_test_operator_name"; |
| const char kOperatorCountry[] = "us"; |
| |
| // Simulate all the side-effects of Cellular::CreateService |
| auto service = |
| new CellularService(&manager_, cellular_->imsi(), cellular_->iccid(), |
| cellular_->GetSimCardId()); |
| service->SetFriendlyName(kFriendlyServiceName); |
| |
| Stringmap serving_operator; |
| serving_operator[kOperatorCodeKey] = kOperatorCode; |
| serving_operator[kOperatorNameKey] = kOperatorName; |
| serving_operator[kOperatorCountryKey] = kOperatorCountry; |
| service->SetServingOperator(serving_operator); |
| cellular_->set_home_provider_for_testing(serving_operator); |
| cellular_->SetServiceForTesting(service); |
| } |
| |
| void ExpectModemAndModem3gppProperties() { |
| modem_properties_.Set<uint32_t>(MM_MODEM_PROPERTY_ACCESSTECHNOLOGIES, |
| kAccessTechnologies); |
| std::tuple<uint32_t, bool> signal_signal{90, true}; |
| modem_properties_.SetVariant(MM_MODEM_PROPERTY_SIGNALQUALITY, |
| brillo::Any(signal_signal)); |
| |
| // Set fake modem 3gpp properties. |
| modem_3gpp_properties_.Set<uint32_t>( |
| MM_MODEM_MODEM3GPP_PROPERTY_ENABLEDFACILITYLOCKS, 0); |
| modem_3gpp_properties_.Set<std::string>(MM_MODEM_MODEM3GPP_PROPERTY_IMEI, |
| kImei); |
| |
| // Set up mock modem signal properties. |
| KeyValueStore modem_signal_property_lte; |
| modem_signal_property_lte.Set<double>( |
| CellularCapability3gpp::kRsrpProperty, |
| CellularCapability3gpp::kRsrpBounds.min_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_LTE, |
| modem_signal_property_lte); |
| } |
| |
| void InvokeEnable(bool enable, |
| Error* error, |
| const ResultCallback& callback, |
| int timeout) { |
| callback.Run(Error()); |
| } |
| void InvokeEnableFail(bool enable, |
| Error* error, |
| const ResultCallback& callback, |
| int timeout) { |
| callback.Run(Error(Error::kOperationFailed)); |
| } |
| void InvokeEnableInWrongState(bool enable, |
| Error* error, |
| const ResultCallback& callback, |
| int timeout) { |
| callback.Run(Error(Error::kWrongState)); |
| } |
| void InvokeSetPowerState(const uint32_t& power_state, |
| Error* error, |
| const ResultCallback& callback, |
| int timeout) { |
| callback.Run(Error()); |
| } |
| |
| void SetSignalProxy() { |
| capability_->modem_signal_proxy_ = std::move(modem_signal_proxy_); |
| } |
| |
| void SetSimpleProxy() { |
| capability_->modem_simple_proxy_ = std::move(modem_simple_proxy_); |
| } |
| |
| void SetMockMobileOperatorInfoObjects() { |
| CHECK(!mock_home_provider_info_); |
| CHECK(!mock_serving_operator_info_); |
| mock_home_provider_info_ = |
| new NiceMock<MockMobileOperatorInfo>(&dispatcher_, "HomeProvider"); |
| mock_serving_operator_info_ = |
| new NiceMock<MockMobileOperatorInfo>(&dispatcher_, "ServingOperator"); |
| mock_home_provider_info_->Init(); |
| mock_serving_operator_info_->Init(); |
| cellular_->set_home_provider_info_for_testing(mock_home_provider_info_); |
| cellular_->set_serving_operator_info_for_testing( |
| mock_serving_operator_info_); |
| } |
| |
| void ReleaseCapabilityProxies() { |
| capability_->ReleaseProxies(); |
| EXPECT_EQ(nullptr, capability_->modem_3gpp_proxy_); |
| EXPECT_EQ(nullptr, capability_->modem_proxy_); |
| EXPECT_EQ(nullptr, capability_->modem_location_proxy_); |
| EXPECT_EQ(nullptr, capability_->modem_signal_proxy_); |
| EXPECT_EQ(nullptr, capability_->modem_simple_proxy_); |
| } |
| |
| void SetRegistrationDroppedUpdateTimeout(int64_t timeout_milliseconds) { |
| capability_->registration_dropped_update_timeout_milliseconds_ = |
| timeout_milliseconds; |
| } |
| |
| void SetMockRegistrationDroppedUpdateCallback() { |
| capability_->registration_dropped_update_callback_.Reset(base::Bind( |
| &CellularCapability3gppTest::FakeCallback, base::Unretained(this))); |
| } |
| |
| void SetApnTryList(const std::deque<Stringmap>& apn) { |
| capability_->apn_try_list_ = apn; |
| } |
| |
| void FillConnectPropertyMap(KeyValueStore* properties) { |
| capability_->FillConnectPropertyMapForTesting(properties); |
| } |
| |
| void CallConnect(const KeyValueStore& properties, |
| const ResultCallback& callback) { |
| capability_->CallConnect(properties, callback); |
| } |
| |
| void StartModem(Error* error) { |
| capability_->StartModem( |
| error, base::Bind(&CellularCapability3gppTest::TestCallback, |
| base::Unretained(this))); |
| } |
| |
| void StopModem(Error* error) { |
| capability_->StopModem(error, |
| base::Bind(&CellularCapability3gppTest::TestCallback, |
| base::Unretained(this))); |
| } |
| |
| void InitProxies() { capability_->InitProxies(); } |
| |
| MOCK_METHOD(void, TestCallback, (const Error&)); |
| MOCK_METHOD(void, FakeCallback, ()); |
| |
| protected: |
| brillo::VariantDictionary GetSimProperties(const RpcIdentifier& sim_path) { |
| const auto iter = std::find(sim_paths_.begin(), sim_paths_.end(), sim_path); |
| if (iter == sim_paths_.end()) |
| return brillo::VariantDictionary(); |
| size_t idx = iter - sim_paths_.begin(); |
| return sim_properties_[idx].properties(); |
| } |
| |
| class TestControl : public MockControl { |
| public: |
| explicit TestControl(CellularCapability3gppTest* test) : test_(test) { |
| active_bearer_properties_.Set<bool>(MM_BEARER_PROPERTY_CONNECTED, true); |
| active_bearer_properties_.Set<std::string>(MM_BEARER_PROPERTY_INTERFACE, |
| "/dev/fake"); |
| |
| KeyValueStore ip4config; |
| ip4config.Set<uint32_t>("method", MM_BEARER_IP_METHOD_DHCP); |
| active_bearer_properties_.Set<KeyValueStore>(MM_BEARER_PROPERTY_IP4CONFIG, |
| ip4config); |
| |
| inactive_bearer_properties_.Set<bool>(MM_BEARER_PROPERTY_CONNECTED, |
| false); |
| } |
| |
| KeyValueStore* mutable_active_bearer_properties() { |
| return &active_bearer_properties_; |
| } |
| |
| KeyValueStore* mutable_inactive_bearer_properties() { |
| return &inactive_bearer_properties_; |
| } |
| |
| std::unique_ptr<mm1::ModemLocationProxyInterface> |
| CreateMM1ModemLocationProxy(const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::make_unique<mm1::MockModemLocationProxy>(); |
| } |
| |
| std::unique_ptr<mm1::ModemModem3gppProxyInterface> |
| CreateMM1ModemModem3gppProxy(const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::move(test_->modem_3gpp_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemProxyInterface> CreateMM1ModemProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::move(test_->modem_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemSignalProxyInterface> CreateMM1ModemSignalProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::move(test_->modem_signal_proxy_); |
| } |
| |
| std::unique_ptr<mm1::ModemSimpleProxyInterface> CreateMM1ModemSimpleProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::move(test_->modem_simple_proxy_); |
| } |
| |
| std::unique_ptr<mm1::SimProxyInterface> CreateMM1SimProxy( |
| const RpcIdentifier& /*path*/, |
| const std::string& /*service*/) override { |
| return std::make_unique<mm1::MockSimProxy>(); |
| } |
| |
| std::unique_ptr<DBusPropertiesProxy> CreateDBusPropertiesProxy( |
| const RpcIdentifier& path, const std::string& /*service*/) override { |
| std::unique_ptr<DBusPropertiesProxy> properties_proxy = |
| DBusPropertiesProxy::CreateDBusPropertiesProxyForTesting( |
| std::make_unique<FakePropertiesProxy>()); |
| FakePropertiesProxy* fake_properties = static_cast<FakePropertiesProxy*>( |
| properties_proxy->GetDBusPropertiesProxyForTesting()); |
| if (path.value().find(kSimPathPrefix) != std::string::npos) { |
| fake_properties->SetDictionaryForTesting(MM_DBUS_INTERFACE_SIM, |
| test_->GetSimProperties(path)); |
| } else if (path.value().find(kActiveBearerPathPrefix) != |
| std::string::npos) { |
| fake_properties->SetDictionaryForTesting( |
| MM_DBUS_INTERFACE_BEARER, active_bearer_properties_.properties()); |
| } else if (path.value().find(kInactiveBearerPathPrefix) != |
| std::string::npos) { |
| fake_properties->SetDictionaryForTesting( |
| MM_DBUS_INTERFACE_BEARER, inactive_bearer_properties_.properties()); |
| } else { |
| fake_properties->SetDictionaryForTesting( |
| MM_DBUS_INTERFACE_MODEM, test_->modem_properties_.properties()); |
| fake_properties->SetForTesting(MM_DBUS_INTERFACE_MODEM, |
| MM_MODEM_PROPERTY_DEVICE, |
| brillo::Any(std::string(kDeviceId))); |
| fake_properties->SetDictionaryForTesting( |
| MM_DBUS_INTERFACE_MODEM_MODEM3GPP, |
| test_->modem_3gpp_properties_.properties()); |
| fake_properties->SetDictionaryForTesting( |
| MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| test_->modem_signal_properties_.properties()); |
| } |
| return properties_proxy; |
| } |
| |
| private: |
| CellularCapability3gppTest* test_; |
| KeyValueStore active_bearer_properties_; |
| KeyValueStore inactive_bearer_properties_; |
| }; |
| |
| EventDispatcherForTest dispatcher_; |
| TestControl control_interface_; |
| NiceMock<MockMetrics> metrics_; |
| NiceMock<MockManager> manager_; |
| NiceMock<MockDeviceInfo> device_info_; |
| MockModemInfo modem_info_; |
| std::unique_ptr<NiceMock<mm1::MockModemModem3gppProxy>> modem_3gpp_proxy_; |
| std::unique_ptr<mm1::MockModemProxy> modem_proxy_; |
| std::unique_ptr<mm1::MockModemSignalProxy> modem_signal_proxy_; |
| std::unique_ptr<mm1::MockModemSimpleProxy> modem_simple_proxy_; |
| CellularCapability3gpp* capability_ = nullptr; // Owned by |cellular_|. |
| DeviceMockAdaptor* device_adaptor_ = nullptr; // Owned by |cellular_|. |
| CellularRefPtr cellular_; |
| MockCellularService* service_ = nullptr; // owned by cellular_ |
| CellularServiceProvider cellular_service_provider_{&manager_}; |
| FakeStore profile_storage_; |
| scoped_refptr<NiceMock<MockProfile>> profile_; |
| |
| // Properties provided by TestControl::CreateDBusPropertiesProxy() |
| KeyValueStore modem_properties_; |
| KeyValueStore modem_3gpp_properties_; |
| KeyValueStore modem_signal_properties_; |
| |
| // saved for testing connect operations. |
| RpcIdentifierCallback connect_callback_; |
| |
| // Set when required and passed to |cellular_|. Owned by |cellular_|. |
| MockMobileOperatorInfo* mock_home_provider_info_; |
| MockMobileOperatorInfo* mock_serving_operator_info_; |
| |
| private: |
| std::vector<RpcIdentifier> sim_paths_; |
| std::vector<KeyValueStore> sim_properties_; |
| }; |
| |
| TEST_F(CellularCapability3gppTest, StartModem) { |
| ExpectModemAndModem3gppProperties(); |
| |
| EXPECT_CALL(*modem_proxy_, |
| Enable(true, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| Error error; |
| StartModem(&error); |
| |
| EXPECT_TRUE(error.IsOngoing()); |
| EXPECT_EQ(kImei, cellular_->imei()); |
| EXPECT_EQ(kAccessTechnologies, |
| capability_->access_technologies_for_testing()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, StartModemFailure) { |
| EXPECT_CALL(*modem_proxy_, |
| Enable(true, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnableFail)); |
| |
| EXPECT_CALL(*this, TestCallback(IsFailure())); |
| Error error; |
| StartModem(&error); |
| EXPECT_TRUE(error.IsOngoing()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, StartModemInWrongState) { |
| ExpectModemAndModem3gppProperties(); |
| |
| EXPECT_CALL(*modem_proxy_, |
| Enable(true, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce( |
| Invoke(this, &CellularCapability3gppTest::InvokeEnableInWrongState)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| |
| EXPECT_CALL(*this, TestCallback(_)).Times(1); |
| Error error; |
| cellular_->set_state_for_testing(Cellular::State::kEnabled); |
| StartModem(&error); |
| EXPECT_TRUE(error.IsOngoing()); |
| |
| // Verify that the modem has not been enabled. |
| EXPECT_TRUE(cellular_->imei().empty()); |
| EXPECT_EQ(0, capability_->access_technologies_for_testing()); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Change the state to kModemStateEnabling and verify that it still has not |
| // been enabled. |
| capability_->OnModemStateChanged(Cellular::kModemStateEnabling); |
| EXPECT_TRUE(cellular_->imei().empty()); |
| EXPECT_EQ(0, capability_->access_technologies_for_testing()); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Change the state to kModemStateDisabling and verify that it still has not |
| // been enabled. |
| EXPECT_CALL(*this, TestCallback(_)).Times(0); |
| capability_->OnModemStateChanged(Cellular::kModemStateDisabling); |
| EXPECT_TRUE(cellular_->imei().empty()); |
| EXPECT_EQ(0, capability_->access_technologies_for_testing()); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Change the state of the modem to disabled and verify that it gets enabled. |
| capability_->OnModemStateChanged(Cellular::kModemStateDisabled); |
| EXPECT_EQ(kImei, cellular_->imei()); |
| EXPECT_EQ(kAccessTechnologies, |
| capability_->access_technologies_for_testing()); |
| Mock::VerifyAndClearExpectations(this); |
| } |
| |
| TEST_F(CellularCapability3gppTest, StopModem) { |
| // Save pointers to proxies before they are lost by the call to InitProxies |
| mm1::MockModemProxy* modem_proxy = modem_proxy_.get(); |
| EXPECT_CALL(*modem_proxy, set_state_changed_callback(_)); |
| InitProxies(); |
| |
| Error error; |
| StopModem(&error); |
| EXPECT_TRUE(error.IsSuccess()); |
| |
| ResultCallback disable_callback; |
| EXPECT_CALL(*modem_proxy, |
| Enable(false, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(SaveArg<2>(&disable_callback)); |
| dispatcher_.DispatchPendingEvents(); |
| |
| ResultCallback set_power_state_callback; |
| EXPECT_CALL( |
| *modem_proxy, |
| SetPowerState(MM_MODEM_POWER_STATE_LOW, _, _, |
| CellularCapability3gpp::kSetPowerStateTimeoutMilliseconds)) |
| .WillOnce(SaveArg<2>(&set_power_state_callback)); |
| disable_callback.Run(Error(Error::kSuccess)); |
| |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| set_power_state_callback.Run(Error(Error::kSuccess)); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // TestCallback should get called with success even if the power state |
| // callback gets called with an error |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| set_power_state_callback.Run(Error(Error::kOperationFailed)); |
| } |
| |
| TEST_F(CellularCapability3gppTest, TerminationAction) { |
| ExpectModemAndModem3gppProperties(); |
| |
| { |
| InSequence seq; |
| |
| EXPECT_CALL(*modem_proxy_, |
| Enable(true, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| EXPECT_CALL(*modem_proxy_, |
| Enable(false, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| EXPECT_CALL(*modem_proxy_, |
| SetPowerState( |
| MM_MODEM_POWER_STATE_LOW, _, _, |
| CellularCapability3gpp::kSetPowerStateTimeoutMilliseconds)) |
| .WillOnce( |
| Invoke(this, &CellularCapability3gppTest::InvokeSetPowerState)); |
| } |
| EXPECT_CALL(*this, TestCallback(IsSuccess())).Times(2); |
| |
| EXPECT_EQ(Cellular::State::kDisabled, cellular_->state()); |
| EXPECT_EQ(Cellular::kModemStateUnknown, cellular_->modem_state()); |
| EXPECT_TRUE(manager_.termination_actions_.IsEmpty()); |
| |
| // Here we mimic the modem state change from ModemManager. When the modem is |
| // enabled, a termination action should be added. |
| cellular_->OnModemStateChanged(Cellular::kModemStateEnabled); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kModemStarted, cellular_->state()); |
| EXPECT_EQ(Cellular::kModemStateEnabled, cellular_->modem_state()); |
| EXPECT_FALSE(manager_.termination_actions_.IsEmpty()); |
| |
| // Running the termination action should disable the modem. |
| manager_.RunTerminationActions(base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this))); |
| dispatcher_.DispatchPendingEvents(); |
| // Here we mimic the modem state change from ModemManager. When the modem is |
| // disabled, the termination action should be removed. |
| cellular_->OnModemStateChanged(Cellular::kModemStateDisabled); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kDisabled, cellular_->state()); |
| EXPECT_EQ(Cellular::kModemStateDisabled, cellular_->modem_state()); |
| EXPECT_TRUE(manager_.termination_actions_.IsEmpty()); |
| |
| // No termination action should be called here. |
| manager_.RunTerminationActions(base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this))); |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularCapability3gppTest, TerminationActionRemovedByStopModem) { |
| ExpectModemAndModem3gppProperties(); |
| |
| { |
| InSequence seq; |
| |
| EXPECT_CALL(*modem_proxy_, |
| Enable(true, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| EXPECT_CALL(*modem_proxy_, |
| Enable(false, _, _, CellularCapability::kTimeoutEnable)) |
| .WillOnce(Invoke(this, &CellularCapability3gppTest::InvokeEnable)); |
| EXPECT_CALL(*modem_proxy_, |
| SetPowerState( |
| MM_MODEM_POWER_STATE_LOW, _, _, |
| CellularCapability3gpp::kSetPowerStateTimeoutMilliseconds)) |
| .WillOnce( |
| Invoke(this, &CellularCapability3gppTest::InvokeSetPowerState)); |
| } |
| EXPECT_CALL(*this, TestCallback(IsSuccess())).Times(1); |
| |
| EXPECT_EQ(Cellular::State::kDisabled, cellular_->state()); |
| EXPECT_EQ(Cellular::kModemStateUnknown, cellular_->modem_state()); |
| EXPECT_TRUE(manager_.termination_actions_.IsEmpty()); |
| |
| // Here we mimic the modem state change from ModemManager. When the modem is |
| // enabled, a termination action should be added. |
| cellular_->OnModemStateChanged(Cellular::kModemStateEnabled); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kModemStarted, cellular_->state()); |
| EXPECT_EQ(Cellular::kModemStateEnabled, cellular_->modem_state()); |
| EXPECT_FALSE(manager_.termination_actions_.IsEmpty()); |
| |
| // Verify that the termination action is removed when the modem is disabled |
| // not due to a suspend request. |
| cellular_->SetEnabled(false); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(Cellular::State::kDisabled, cellular_->state()); |
| EXPECT_TRUE(manager_.termination_actions_.IsEmpty()); |
| |
| // No termination action should be called here. |
| manager_.RunTerminationActions(base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this))); |
| dispatcher_.DispatchPendingEvents(); |
| } |
| |
| TEST_F(CellularCapability3gppTest, DisconnectModemNoBearer) { |
| ResultCallback disconnect_callback; |
| EXPECT_CALL(*modem_simple_proxy_, |
| Disconnect(_, _, CellularCapability::kTimeoutDisconnect)) |
| .Times(0); |
| capability_->Disconnect(disconnect_callback); |
| } |
| |
| TEST_F(CellularCapability3gppTest, DisconnectNoProxy) { |
| ResultCallback disconnect_callback; |
| EXPECT_CALL(*modem_simple_proxy_, |
| Disconnect(_, _, CellularCapability::kTimeoutDisconnect)) |
| .Times(0); |
| ReleaseCapabilityProxies(); |
| capability_->Disconnect(disconnect_callback); |
| } |
| |
| TEST_F(CellularCapability3gppTest, SimLockStatusChanged) { |
| InitProxies(); |
| |
| // Set up mock SIM properties |
| const char kSimIdentifier[] = "9999888"; |
| const char kOperatorIdentifier[] = "310240"; |
| const char kOperatorName[] = "Custom SPN"; |
| KeyValueStore sim_properties; |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_IMSI, kImsi); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, |
| kSimIdentifier); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_OPERATORIDENTIFIER, |
| kOperatorIdentifier); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_OPERATORNAME, kOperatorName); |
| |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| |
| SetSimPropertiesAndPath(kSimPath1, sim_properties); |
| |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_NE(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| |
| ClearCellularSimProperties(); |
| |
| // SIM is locked. |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PIN; |
| capability_->OnSimLockStatusChanged(); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| |
| // SIM is unlocked. |
| SetSimPropertiesAndPath(kSimPath1, sim_properties); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_NONE; |
| capability_->OnSimLockStatusChanged(); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_EQ(kImsi, cellular_->imsi()); |
| EXPECT_EQ(kSimIdentifier, cellular_->iccid()); |
| |
| // SIM is missing and SIM path is "/". |
| ClearCapabilitySimProperties(); |
| SetSimPath(CellularCapability3gpp::kRootPath); |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(CellularCapability3gpp::kRootPath, |
| capability_->sim_path_for_testing()); |
| |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(_, _)) |
| .Times(0); |
| |
| capability_->OnSimLockStatusChanged(); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| |
| // SIM is missing and SIM path is empty. |
| ClearCapabilitySimProperties(); |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(RpcIdentifier(""), capability_->sim_path_for_testing()); |
| |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(_, _)) |
| .Times(0); |
| capability_->OnSimLockStatusChanged(); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, PropertiesChanged) { |
| InitProxies(); |
| |
| // Set up mock modem properties |
| KeyValueStore modem_properties; |
| modem_properties.Set<uint32_t>(MM_MODEM_PROPERTY_ACCESSTECHNOLOGIES, |
| kAccessTechnologies); |
| modem_properties.Set<RpcIdentifier>(MM_MODEM_PROPERTY_SIM, kSimPath1); |
| |
| // Set up mock modem 3gpp properties |
| KeyValueStore modem3gpp_properties; |
| modem3gpp_properties.Set<uint32_t>( |
| MM_MODEM_MODEM3GPP_PROPERTY_ENABLEDFACILITYLOCKS, 0); |
| modem3gpp_properties.Set<std::string>(MM_MODEM_MODEM3GPP_PROPERTY_IMEI, |
| kImei); |
| |
| // Set up mock modem sim properties |
| SetSimPropertiesAndPath(CellularCapability3gpp::kRootPath, {}); |
| |
| EXPECT_EQ("", cellular_->imei()); |
| EXPECT_EQ(MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN, |
| capability_->access_technologies_for_testing()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| // Cellular::SetPrimarySimProperties will emit Cellular properties. |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(_, _)).Times(AnyNumber()); |
| // Ensure that Family and Imei are set properly. |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(kTechnologyFamilyProperty, |
| kTechnologyFamilyGsm)) |
| .Times(1); |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(kImeiProperty, kImei)) |
| .Times(1); |
| |
| SetSimPropertiesAndPath(kSimPath1, {}); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| dispatcher_.DispatchPendingEvents(); |
| |
| EXPECT_EQ(kAccessTechnologies, |
| capability_->access_technologies_for_testing()); |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_NE(nullptr, capability_->sim_proxy_); |
| |
| // Changing properties on wrong interface will not have an effect |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, |
| modem3gpp_properties); |
| EXPECT_EQ("", cellular_->imei()); |
| |
| // Changing properties on the right interface gets reflected in the |
| // capabilities object |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_MODEM3GPP, |
| modem3gpp_properties); |
| EXPECT_EQ(kImei, cellular_->imei()); |
| Mock::VerifyAndClearExpectations(device_adaptor_); |
| |
| // Expect to see changes when the family changes |
| modem_properties.Clear(); |
| modem_properties.Set<uint32_t>(MM_MODEM_PROPERTY_ACCESSTECHNOLOGIES, |
| MM_MODEM_ACCESS_TECHNOLOGY_1XRTT); |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(kTechnologyFamilyProperty, |
| kTechnologyFamilyCdma)) |
| .Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| Mock::VerifyAndClearExpectations(device_adaptor_); |
| |
| // Back to LTE |
| modem_properties.Clear(); |
| modem_properties.Set<uint32_t>(MM_MODEM_PROPERTY_ACCESSTECHNOLOGIES, |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE); |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(kTechnologyFamilyProperty, |
| kTechnologyFamilyGsm)) |
| .Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| Mock::VerifyAndClearExpectations(device_adaptor_); |
| |
| // LTE & CDMA - the device adaptor should not be called! |
| modem_properties.Clear(); |
| modem_properties.Set<uint32_t>( |
| MM_MODEM_PROPERTY_ACCESSTECHNOLOGIES, |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE | MM_MODEM_ACCESS_TECHNOLOGY_1XRTT); |
| EXPECT_CALL(*device_adaptor_, EmitStringChanged(_, _)).Times(0); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| } |
| |
| TEST_F(CellularCapability3gppTest, SignalPropertiesChanged) { |
| modem_signal_properties_.Clear(); |
| |
| KeyValueStore modem_signal_property_gsm; |
| modem_signal_property_gsm.Set<double>( |
| CellularCapability3gpp::kRssiProperty, |
| CellularCapability3gpp::kRssiBounds.max_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_GSM, |
| modem_signal_property_gsm); |
| EXPECT_CALL(*service_, SetStrength(100)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| KeyValueStore modem_signal_property_umts; |
| modem_signal_property_umts.Set<double>( |
| CellularCapability3gpp::kRssiProperty, |
| CellularCapability3gpp::kRssiBounds.min_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_UMTS, |
| modem_signal_property_umts); |
| EXPECT_CALL(*service_, SetStrength(0)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| modem_signal_property_umts.Set<double>( |
| CellularCapability3gpp::kRscpProperty, |
| CellularCapability3gpp::kRscpBounds.min_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_UMTS, |
| modem_signal_property_umts); |
| EXPECT_CALL(*service_, SetStrength(0)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| modem_signal_property_umts.Set<double>( |
| CellularCapability3gpp::kRscpProperty, |
| CellularCapability3gpp::kRscpBounds.max_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_UMTS, |
| modem_signal_property_umts); |
| EXPECT_CALL(*service_, SetStrength(100)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| double rscp_midrange = (CellularCapability3gpp::kRscpBounds.min_threshold + |
| CellularCapability3gpp::kRscpBounds.max_threshold) / |
| 2; |
| modem_signal_property_umts.Set<double>(CellularCapability3gpp::kRscpProperty, |
| rscp_midrange); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_UMTS, |
| modem_signal_property_umts); |
| uint32_t expected_strength_rscp = |
| CellularCapability3gpp::kRscpBounds.GetAsPercentage(rscp_midrange); |
| EXPECT_CALL(*service_, SetStrength(expected_strength_rscp)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| KeyValueStore modem_signal_property_lte; |
| modem_signal_property_lte.Set<double>( |
| CellularCapability3gpp::kRssiProperty, |
| CellularCapability3gpp::kRssiBounds.max_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_LTE, |
| modem_signal_property_lte); |
| EXPECT_CALL(*service_, SetStrength(100)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| modem_signal_property_lte.Set<double>( |
| CellularCapability3gpp::kRsrpProperty, |
| CellularCapability3gpp::kRsrpBounds.min_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_LTE, |
| modem_signal_property_lte); |
| EXPECT_CALL(*service_, SetStrength(0)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| modem_signal_property_lte.Set<double>( |
| CellularCapability3gpp::kRsrpProperty, |
| CellularCapability3gpp::kRsrpBounds.max_threshold); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_LTE, |
| modem_signal_property_lte); |
| EXPECT_CALL(*service_, SetStrength(100)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| |
| double rsrp_midrange = (CellularCapability3gpp::kRsrpBounds.min_threshold + |
| CellularCapability3gpp::kRsrpBounds.max_threshold) / |
| 2; |
| modem_signal_property_lte.Set<double>(CellularCapability3gpp::kRsrpProperty, |
| rsrp_midrange); |
| modem_signal_properties_.Set<KeyValueStore>(MM_MODEM_SIGNAL_PROPERTY_LTE, |
| modem_signal_property_lte); |
| uint32_t expected_strength = |
| CellularCapability3gpp::kRsrpBounds.GetAsPercentage(rsrp_midrange); |
| EXPECT_CALL(*service_, SetStrength(expected_strength)).Times(1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM_SIGNAL, |
| modem_signal_properties_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdateRegistrationState) { |
| InitProxies(); |
| |
| CreateService(); |
| SetDefaultCellularSimProperties(); |
| cellular_->set_modem_state_for_testing(Cellular::kModemStateConnected); |
| SetRegistrationDroppedUpdateTimeout(0); |
| |
| const Stringmap& home_provider_map = cellular_->home_provider(); |
| ASSERT_NE(home_provider_map.end(), home_provider_map.find(kOperatorNameKey)); |
| std::string home_provider = home_provider_map.find(kOperatorNameKey)->second; |
| std::string ota_name = cellular_->service()->friendly_name(); |
| |
| // Home --> Roaming should be effective immediately. |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_ROAMING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_ROAMING, |
| capability_->registration_state_); |
| |
| // Idle --> Roaming should be effective immediately. |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_IDLE, |
| home_provider, ota_name); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_IDLE, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_ROAMING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_ROAMING, |
| capability_->registration_state_); |
| |
| // Idle --> Searching should be effective immediately. |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_IDLE, |
| home_provider, ota_name); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_IDLE, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, |
| capability_->registration_state_); |
| |
| // Home --> Searching --> Home should never see Searching. |
| EXPECT_CALL(metrics_, Notify3GPPRegistrationDelayedDropPosted()); |
| EXPECT_CALL(metrics_, Notify3GPPRegistrationDelayedDropCanceled()); |
| |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| Mock::VerifyAndClearExpectations(&metrics_); |
| |
| // Home --> Searching --> wait till dispatch should see Searching |
| EXPECT_CALL(metrics_, Notify3GPPRegistrationDelayedDropPosted()); |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, |
| capability_->registration_state_); |
| Mock::VerifyAndClearExpectations(&metrics_); |
| |
| // Home --> Searching --> Searching --> wait till dispatch should see |
| // Searching *and* the first callback should be cancelled. |
| EXPECT_CALL(*this, FakeCallback()).Times(0); |
| EXPECT_CALL(metrics_, Notify3GPPRegistrationDelayedDropPosted()); |
| |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| SetMockRegistrationDroppedUpdateCallback(); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| dispatcher_.DispatchPendingEvents(); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, |
| capability_->registration_state_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, IsRegistered) { |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_IDLE; |
| EXPECT_FALSE(capability_->IsRegistered()); |
| |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_HOME; |
| EXPECT_TRUE(capability_->IsRegistered()); |
| |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING; |
| EXPECT_FALSE(capability_->IsRegistered()); |
| |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_DENIED; |
| EXPECT_FALSE(capability_->IsRegistered()); |
| |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_UNKNOWN; |
| EXPECT_FALSE(capability_->IsRegistered()); |
| |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_ROAMING; |
| EXPECT_TRUE(capability_->IsRegistered()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdateRegistrationStateModemNotConnected) { |
| InitProxies(); |
| CreateService(); |
| |
| SetDefaultCellularSimProperties(); |
| cellular_->set_modem_state_for_testing(Cellular::kModemStateRegistered); |
| SetRegistrationDroppedUpdateTimeout(0); |
| |
| const Stringmap& home_provider_map = cellular_->home_provider(); |
| ASSERT_NE(home_provider_map.end(), home_provider_map.find(kOperatorNameKey)); |
| std::string home_provider = home_provider_map.find(kOperatorNameKey)->second; |
| std::string ota_name = cellular_->service()->friendly_name(); |
| |
| // Home --> Searching should be effective immediately. |
| capability_->On3gppRegistrationChanged(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_HOME, |
| capability_->registration_state_); |
| capability_->On3gppRegistrationChanged( |
| MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, home_provider, ota_name); |
| EXPECT_EQ(MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING, |
| capability_->registration_state_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, IsValidSimPath) { |
| // Invalid paths |
| EXPECT_FALSE(capability_->IsValidSimPath(RpcIdentifier(""))); |
| EXPECT_FALSE(capability_->IsValidSimPath(RpcIdentifier("/"))); |
| |
| // A valid path |
| EXPECT_TRUE(capability_->IsValidSimPath( |
| RpcIdentifier("/org/freedesktop/ModemManager1/SIM/0"))); |
| |
| // Note that any string that is not one of the above invalid paths is |
| // currently regarded as valid, since the ModemManager spec doesn't impose |
| // a strict format on the path. The validity of this is subject to change. |
| EXPECT_TRUE(capability_->IsValidSimPath(RpcIdentifier("path"))); |
| } |
| |
| TEST_F(CellularCapability3gppTest, NormalizeMdn) { |
| EXPECT_EQ("", capability_->NormalizeMdn("")); |
| EXPECT_EQ("12345678901", capability_->NormalizeMdn("12345678901")); |
| EXPECT_EQ("12345678901", capability_->NormalizeMdn("+1 234 567 8901")); |
| EXPECT_EQ("12345678901", capability_->NormalizeMdn("+1-234-567-8901")); |
| EXPECT_EQ("12345678901", capability_->NormalizeMdn("+1 (234) 567-8901")); |
| EXPECT_EQ("12345678901", capability_->NormalizeMdn("1 234 567 8901 ")); |
| EXPECT_EQ("2345678901", capability_->NormalizeMdn("(234) 567-8901")); |
| } |
| |
| TEST_F(CellularCapability3gppTest, SimPathChanged) { |
| InitProxies(); |
| |
| // Set up mock modem SIM properties |
| const char kSimIdentifier[] = "9999888"; |
| const char kOperatorIdentifier[] = "310240"; |
| const char kOperatorName[] = "Custom SPN"; |
| KeyValueStore sim_properties; |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_IMSI, kImsi); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, |
| kSimIdentifier); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_OPERATORIDENTIFIER, |
| kOperatorIdentifier); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_OPERATORNAME, kOperatorName); |
| |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(RpcIdentifier(""), capability_->sim_path_for_testing()); |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| |
| SetSimPropertiesAndPath(kSimPath1, sim_properties); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_NE(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_EQ(kImsi, cellular_->imsi()); |
| EXPECT_EQ(kSimIdentifier, cellular_->iccid()); |
| |
| // Changing to the same SIM path should be a no-op. |
| SetSimPath(kSimPath1); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_NE(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_EQ(kImsi, cellular_->imsi()); |
| EXPECT_EQ(kSimIdentifier, cellular_->iccid()); |
| |
| // SIM is removed, Modem.Sim path is empty. |
| ClearCapabilitySimProperties(); |
| VerifyAndSetActivationExpectations(); |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(RpcIdentifier(""), capability_->sim_path_for_testing()); |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| |
| // SIM is replaced. |
| SetSimPropertiesAndPath(kSimPath1, sim_properties); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_NE(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_EQ(kImsi, cellular_->imsi()); |
| EXPECT_EQ(kSimIdentifier, cellular_->iccid()); |
| |
| // SIM is removed, Modem.Sim path is "/". |
| ClearCapabilitySimProperties(); |
| SetSimPath(CellularCapability3gpp::kRootPath); |
| EXPECT_FALSE(cellular_->sim_present()); |
| EXPECT_EQ(nullptr, capability_->sim_proxy_); |
| EXPECT_EQ(CellularCapability3gpp::kRootPath, |
| capability_->sim_path_for_testing()); |
| EXPECT_EQ("", cellular_->imsi()); |
| EXPECT_EQ("", cellular_->iccid()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, Reset) { |
| // Save pointers to proxies before they are lost by the call to InitProxies |
| mm1::MockModemProxy* modem_proxy = modem_proxy_.get(); |
| EXPECT_CALL(*modem_proxy, set_state_changed_callback(_)); |
| InitProxies(); |
| |
| Error error; |
| ResultCallback reset_callback; |
| |
| EXPECT_CALL(*modem_proxy, Reset(_, _, CellularCapability::kTimeoutReset)) |
| .WillOnce(SaveArg<1>(&reset_callback)); |
| |
| capability_->Reset(&error, ResultCallback()); |
| EXPECT_TRUE(capability_->resetting_); |
| reset_callback.Run(error); |
| EXPECT_FALSE(capability_->resetting_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdateActiveBearer) { |
| // Common resources. |
| const size_t kPathCount = 3; |
| RpcIdentifier active_paths[kPathCount], inactive_paths[kPathCount]; |
| for (size_t i = 0; i < kPathCount; ++i) { |
| active_paths[i] = |
| RpcIdentifier(base::StringPrintf("%s/%zu", kActiveBearerPathPrefix, i)); |
| inactive_paths[i] = RpcIdentifier( |
| base::StringPrintf("%s/%zu", kInactiveBearerPathPrefix, i)); |
| } |
| |
| EXPECT_EQ(nullptr, capability_->GetActiveBearer()); |
| |
| // Check that |active_bearer_| is set correctly when an active bearer is |
| // returned. |
| capability_->OnBearersChanged({inactive_paths[0], inactive_paths[1], |
| active_paths[2], inactive_paths[1], |
| inactive_paths[2]}); |
| capability_->UpdateActiveBearer(); |
| ASSERT_NE(nullptr, capability_->GetActiveBearer()); |
| EXPECT_EQ(active_paths[2], capability_->GetActiveBearer()->dbus_path()); |
| |
| // Check that |active_bearer_| is nullptr if no active bearers are returned. |
| capability_->OnBearersChanged({inactive_paths[0], inactive_paths[1], |
| inactive_paths[2], inactive_paths[1]}); |
| capability_->UpdateActiveBearer(); |
| EXPECT_EQ(nullptr, capability_->GetActiveBearer()); |
| |
| // Check that returning multiple bearers causes death. |
| capability_->OnBearersChanged({active_paths[0], inactive_paths[1], |
| inactive_paths[2], active_paths[1], |
| inactive_paths[1]}); |
| EXPECT_DEATH(capability_->UpdateActiveBearer(), |
| "Found more than one active bearer."); |
| |
| capability_->OnBearersChanged({}); |
| capability_->UpdateActiveBearer(); |
| EXPECT_EQ(nullptr, capability_->GetActiveBearer()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, SetInitialEpsBearer) { |
| constexpr char kTestApn[] = "test_apn"; |
| KeyValueStore properties; |
| Error error; |
| ResultCallback callback = base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this)); |
| |
| ResultCallback set_callback; |
| EXPECT_CALL(*modem_3gpp_proxy_, |
| SetInitialEpsBearerSettings( |
| _, _, _, CellularCapability::kTimeoutSetInitialEpsBearer)) |
| .Times(1) |
| .WillOnce(SaveArg<2>(&set_callback)); |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| properties.Set<std::string>(CellularCapability3gpp::kConnectApn, kTestApn); |
| |
| cellular_->set_use_attach_apn_for_testing(true); |
| |
| InitProxies(); |
| capability_->SetInitialEpsBearer(properties, &error, callback); |
| set_callback.Run(Error(Error::kSuccess)); |
| } |
| |
| // Validates FillConnectPropertyMap |
| TEST_F(CellularCapability3gppTest, FillConnectPropertyMap) { |
| constexpr char kTestApn[] = "test_apn"; |
| constexpr char kTestUser[] = "test_user"; |
| constexpr char kTestPassword[] = "test_password"; |
| |
| KeyValueStore properties; |
| Stringmap apn; |
| apn[kApnProperty] = kTestApn; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasNoUser()); |
| EXPECT_THAT(properties, HasNoPassword()); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnUsernameProperty] = kTestUser; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasNoPassword()); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnPasswordProperty] = kTestPassword; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnAuthenticationProperty] = kApnAuthenticationPap; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasAllowedAuth(MM_BEARER_ALLOWED_AUTH_PAP)); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnAuthenticationProperty] = kApnAuthenticationChap; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasAllowedAuth(MM_BEARER_ALLOWED_AUTH_CHAP)); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnAuthenticationProperty] = "something"; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnAuthenticationProperty] = ""; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasNoIpType()); |
| |
| apn[kApnIpTypeProperty] = kApnIpTypeV4; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasIpType(MM_BEARER_IP_FAMILY_IPV4)); |
| |
| apn[kApnIpTypeProperty] = kApnIpTypeV6; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasIpType(MM_BEARER_IP_FAMILY_IPV6)); |
| |
| apn[kApnIpTypeProperty] = kApnIpTypeV4V6; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasIpType(MM_BEARER_IP_FAMILY_IPV4V6)); |
| |
| // IP type defaults to v4 if something unsupported is specified. |
| apn[kApnIpTypeProperty] = "orekid"; |
| SetApnTryList({apn}); |
| FillConnectPropertyMap(&properties); |
| EXPECT_THAT(properties, HasApn(kTestApn)); |
| EXPECT_THAT(properties, HasUser(kTestUser)); |
| EXPECT_THAT(properties, HasPassword(kTestPassword)); |
| EXPECT_THAT(properties, HasNoAllowedAuth()); |
| EXPECT_THAT(properties, HasIpType(MM_BEARER_IP_FAMILY_IPV4)); |
| } |
| |
| // Validates expected behavior of Connect function |
| TEST_F(CellularCapability3gppTest, Connect) { |
| mm1::MockModemSimpleProxy* modem_simple_proxy = modem_simple_proxy_.get(); |
| SetSimpleProxy(); |
| SetApnTryList({}); |
| ResultCallback callback = base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this)); |
| RpcIdentifier bearer("/foo"); |
| |
| // Test connect failures |
| EXPECT_CALL(*modem_simple_proxy, Connect(_, _, _)) |
| .WillRepeatedly(SaveArg<1>(&connect_callback_)); |
| capability_->Connect(callback); |
| EXPECT_CALL(*this, TestCallback(IsFailure())); |
| EXPECT_CALL(*service_, ClearLastGoodApn()); |
| connect_callback_.Run(bearer, Error(Error::kOperationFailed)); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Test connect success |
| capability_->Connect(callback); |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| connect_callback_.Run(bearer, Error(Error::kSuccess)); |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Test connect failures without a service. Make sure that shill |
| // does not crash if the connect failed and there is no |
| // CellularService object. This can happen if the modem is enabled |
| // and then quickly disabled. |
| cellular_->SetServiceForTesting(nullptr); |
| EXPECT_FALSE(capability_->cellular()->service()); |
| capability_->Connect(callback); |
| EXPECT_CALL(*this, TestCallback(IsFailure())); |
| connect_callback_.Run(bearer, Error(Error::kOperationFailed)); |
| } |
| |
| // Validates Connect iterates over APNs |
| TEST_F(CellularCapability3gppTest, ConnectApns) { |
| mm1::MockModemSimpleProxy* modem_simple_proxy = modem_simple_proxy_.get(); |
| SetSimpleProxy(); |
| KeyValueStore properties; |
| ResultCallback callback = base::Bind( |
| &CellularCapability3gppTest::TestCallback, base::Unretained(this)); |
| RpcIdentifier bearer("/bearer0"); |
| |
| const char apn_name_foo[] = "foo"; |
| const char apn_name_bar[] = "bar"; |
| EXPECT_CALL(*modem_simple_proxy, Connect(HasApn(apn_name_foo), _, _)) |
| .WillOnce(SaveArg<1>(&connect_callback_)); |
| Stringmap apn1; |
| apn1[kApnProperty] = apn_name_foo; |
| Stringmap apn2; |
| apn2[kApnProperty] = apn_name_bar; |
| SetApnTryList({apn1, apn2}); |
| FillConnectPropertyMap(&properties); |
| CallConnect(properties, callback); |
| Mock::VerifyAndClearExpectations(modem_simple_proxy); |
| |
| EXPECT_CALL(*modem_simple_proxy, Connect(HasApn(apn_name_bar), _, _)) |
| .WillOnce(SaveArg<1>(&connect_callback_)); |
| EXPECT_CALL(*service_, ClearLastGoodApn()); |
| connect_callback_.Run(bearer, Error(Error::kInvalidApn)); |
| |
| EXPECT_CALL(*service_, SetLastGoodApn(apn2)); |
| EXPECT_CALL(*this, TestCallback(IsSuccess())); |
| connect_callback_.Run(bearer, Error(Error::kSuccess)); |
| } |
| |
| // Validates GetTypeString and AccessTechnologyToTechnologyFamily |
| TEST_F(CellularCapability3gppTest, GetTypeString) { |
| static const uint32_t kGsmTechnologies[] = { |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE, |
| MM_MODEM_ACCESS_TECHNOLOGY_HSPA_PLUS, |
| MM_MODEM_ACCESS_TECHNOLOGY_HSPA, |
| MM_MODEM_ACCESS_TECHNOLOGY_HSUPA, |
| MM_MODEM_ACCESS_TECHNOLOGY_HSDPA, |
| MM_MODEM_ACCESS_TECHNOLOGY_UMTS, |
| MM_MODEM_ACCESS_TECHNOLOGY_EDGE, |
| MM_MODEM_ACCESS_TECHNOLOGY_GPRS, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM_COMPACT, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM, |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE | MM_MODEM_ACCESS_TECHNOLOGY_EVDO0, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM | MM_MODEM_ACCESS_TECHNOLOGY_EVDO0, |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE | MM_MODEM_ACCESS_TECHNOLOGY_EVDOA, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM | MM_MODEM_ACCESS_TECHNOLOGY_EVDOA, |
| MM_MODEM_ACCESS_TECHNOLOGY_LTE | MM_MODEM_ACCESS_TECHNOLOGY_EVDOB, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM | MM_MODEM_ACCESS_TECHNOLOGY_EVDOB, |
| MM_MODEM_ACCESS_TECHNOLOGY_GSM | MM_MODEM_ACCESS_TECHNOLOGY_1XRTT, |
| }; |
| for (auto gsm_technology : kGsmTechnologies) { |
| capability_->access_technologies_ = gsm_technology; |
| ASSERT_EQ(capability_->GetTypeString(), kTechnologyFamilyGsm); |
| } |
| static const uint32_t kCdmaTechnologies[] = { |
| MM_MODEM_ACCESS_TECHNOLOGY_EVDO0, |
| MM_MODEM_ACCESS_TECHNOLOGY_EVDOA, |
| MM_MODEM_ACCESS_TECHNOLOGY_EVDOA | MM_MODEM_ACCESS_TECHNOLOGY_EVDO0, |
| MM_MODEM_ACCESS_TECHNOLOGY_EVDOB, |
| MM_MODEM_ACCESS_TECHNOLOGY_EVDOB | MM_MODEM_ACCESS_TECHNOLOGY_EVDO0, |
| MM_MODEM_ACCESS_TECHNOLOGY_1XRTT, |
| }; |
| for (auto cdma_technology : kCdmaTechnologies) { |
| capability_->access_technologies_ = cdma_technology; |
| ASSERT_EQ(capability_->GetTypeString(), kTechnologyFamilyCdma); |
| } |
| capability_->access_technologies_ = MM_MODEM_ACCESS_TECHNOLOGY_UNKNOWN; |
| ASSERT_EQ(capability_->GetTypeString(), ""); |
| } |
| |
| TEST_F(CellularCapability3gppTest, GetMdnForOLP) { |
| const std::string kVzwUUID = "c83d6597-dc91-4d48-a3a7-d86b80123751"; |
| const std::string kFooUUID = "foo"; |
| MockMobileOperatorInfo mock_operator_info(&dispatcher_, "MobileOperatorInfo"); |
| |
| EXPECT_CALL(mock_operator_info, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(mock_operator_info, uuid()).WillRepeatedly(ReturnRef(kVzwUUID)); |
| capability_->subscription_state_ = SubscriptionState::kUnknown; |
| |
| cellular_->SetMdn(""); |
| EXPECT_EQ("0000000000", capability_->GetMdnForOLP(&mock_operator_info)); |
| cellular_->SetMdn("0123456789"); |
| EXPECT_EQ("0123456789", capability_->GetMdnForOLP(&mock_operator_info)); |
| cellular_->SetMdn("10123456789"); |
| EXPECT_EQ("0123456789", capability_->GetMdnForOLP(&mock_operator_info)); |
| |
| cellular_->SetMdn("1021232333"); |
| capability_->subscription_state_ = SubscriptionState::kUnprovisioned; |
| EXPECT_EQ("0000000000", capability_->GetMdnForOLP(&mock_operator_info)); |
| Mock::VerifyAndClearExpectations(&mock_operator_info); |
| |
| EXPECT_CALL(mock_operator_info, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(mock_operator_info, uuid()).WillRepeatedly(ReturnRef(kFooUUID)); |
| |
| cellular_->SetMdn(""); |
| EXPECT_EQ("", capability_->GetMdnForOLP(&mock_operator_info)); |
| cellular_->SetMdn("0123456789"); |
| EXPECT_EQ("0123456789", capability_->GetMdnForOLP(&mock_operator_info)); |
| cellular_->SetMdn("10123456789"); |
| EXPECT_EQ("10123456789", capability_->GetMdnForOLP(&mock_operator_info)); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdateServiceOLP) { |
| const MobileOperatorInfo::OnlinePortal kOlp{ |
| "http://testurl", "POST", |
| "imei=${imei}&imsi=${imsi}&mdn=${mdn}&min=${min}&iccid=${iccid}"}; |
| const std::vector<MobileOperatorInfo::OnlinePortal> kOlpList{kOlp}; |
| const std::string kUuidVzw = "c83d6597-dc91-4d48-a3a7-d86b80123751"; |
| const std::string kUuidFoo = "foo"; |
| |
| cellular_->SetImei("1"); |
| Cellular::SimProperties sim_properties; |
| sim_properties.iccid = "6"; |
| sim_properties.imsi = "2"; |
| SetCellularSimProperties(sim_properties); |
| cellular_->SetMdn("10123456789"); |
| cellular_->SetMin("5"); |
| |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_home_provider_info_, olp_list()) |
| .WillRepeatedly(ReturnRef(kOlpList)); |
| EXPECT_CALL(*mock_home_provider_info_, uuid()) |
| .WillRepeatedly(ReturnRef(kUuidVzw)); |
| CreateService(); |
| capability_->UpdateServiceOLP(); |
| // Copy to simplify assertions below. |
| Stringmap vzw_olp = cellular_->service()->olp(); |
| EXPECT_EQ("http://testurl", vzw_olp[kPaymentPortalURL]); |
| EXPECT_EQ("POST", vzw_olp[kPaymentPortalMethod]); |
| EXPECT_EQ("imei=1&imsi=2&mdn=0123456789&min=5&iccid=6", |
| vzw_olp[kPaymentPortalPostData]); |
| Mock::VerifyAndClearExpectations(mock_home_provider_info_); |
| |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_home_provider_info_, olp_list()) |
| .WillRepeatedly(ReturnRef(kOlpList)); |
| EXPECT_CALL(*mock_home_provider_info_, uuid()) |
| .WillRepeatedly(ReturnRef(kUuidFoo)); |
| capability_->UpdateServiceOLP(); |
| // Copy to simplify assertions below. |
| Stringmap olp = cellular_->service()->olp(); |
| EXPECT_EQ("http://testurl", olp[kPaymentPortalURL]); |
| EXPECT_EQ("POST", olp[kPaymentPortalMethod]); |
| EXPECT_EQ("imei=1&imsi=2&mdn=10123456789&min=5&iccid=6", |
| olp[kPaymentPortalPostData]); |
| } |
| |
| TEST_F(CellularCapability3gppTest, IsMdnValid) { |
| cellular_->SetMdn(""); |
| EXPECT_FALSE(capability_->IsMdnValid()); |
| cellular_->SetMdn("0000000"); |
| EXPECT_FALSE(capability_->IsMdnValid()); |
| cellular_->SetMdn("0000001"); |
| EXPECT_TRUE(capability_->IsMdnValid()); |
| cellular_->SetMdn("1231223"); |
| EXPECT_TRUE(capability_->IsMdnValid()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, CompleteActivation) { |
| SetDefaultCellularSimProperties(); |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| SetActivationState(PendingActivationStore::kIdentifierICCID, |
| kIccid, PendingActivationStore::kStatePending)) |
| .Times(1); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .WillOnce(Return(PendingActivationStore::kStatePending)); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivating)) |
| .Times(1); |
| EXPECT_CALL(*modem_proxy_, Reset(_, _, _)).Times(1); |
| Error error; |
| InitProxies(); |
| capability_->CompleteActivation(&error); |
| VerifyAndSetActivationExpectations(); |
| Mock::VerifyAndClearExpectations(service_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdateServiceActivationState) { |
| const std::vector<MobileOperatorInfo::OnlinePortal> olp_list{ |
| {"some@url", "some_method", "some_post_data"}}; |
| |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .WillRepeatedly(Return(PendingActivationStore::kStateUnknown)); |
| |
| capability_->subscription_state_ = SubscriptionState::kUnprovisioned; |
| ClearCellularSimProperties(); |
| cellular_->SetMdn("0000000000"); |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_home_provider_info_, olp_list()) |
| .WillRepeatedly(ReturnRef(olp_list)); |
| |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateNotActivated)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| |
| cellular_->SetMdn("1231231122"); |
| capability_->subscription_state_ = SubscriptionState::kUnknown; |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivated)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| |
| cellular_->SetMdn("0000000000"); |
| SetDefaultCellularSimProperties(); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .Times(1) |
| .WillRepeatedly(Return(PendingActivationStore::kStatePending)); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivating)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .Times(2) |
| .WillRepeatedly(Return(PendingActivationStore::kStateActivated)); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivated)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .WillRepeatedly(Return(PendingActivationStore::kStateUnknown)); |
| |
| // SubscriptionStateUnprovisioned overrides valid MDN. |
| capability_->subscription_state_ = SubscriptionState::kUnprovisioned; |
| cellular_->SetMdn("1231231122"); |
| ClearCellularSimProperties(); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateNotActivated)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| |
| // SubscriptionStateProvisioned overrides invalid MDN. |
| capability_->subscription_state_ = SubscriptionState::kProvisioned; |
| cellular_->SetMdn("0000000000"); |
| ClearCellularSimProperties(); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivated)) |
| .Times(1); |
| capability_->UpdateServiceActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| } |
| |
| TEST_F(CellularCapability3gppTest, UpdatePendingActivationState) { |
| InitProxies(); |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING; |
| |
| // No MDN, no ICCID. |
| cellular_->SetMdn("0000000"); |
| capability_->subscription_state_ = SubscriptionState::kUnknown; |
| ClearCellularSimProperties(); |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .Times(0); |
| capability_->UpdatePendingActivationState(); |
| VerifyAndSetActivationExpectations(); |
| |
| // Valid MDN, but subsciption_state_ Unprovisioned |
| cellular_->SetMdn("1234567"); |
| capability_->subscription_state_ = SubscriptionState::kUnprovisioned; |
| ClearCellularSimProperties(); |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, _)) |
| .Times(0); |
| capability_->UpdatePendingActivationState(); |
| VerifyAndSetActivationExpectations(); |
| |
| // ICCID known. |
| SetDefaultCellularSimProperties(); |
| |
| // After the modem has reset. |
| capability_->reset_done_ = true; |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .Times(1) |
| .WillOnce(Return(PendingActivationStore::kStatePending)); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| SetActivationState(PendingActivationStore::kIdentifierICCID, kIccid, |
| PendingActivationStore::kStateActivated)) |
| .Times(1); |
| EXPECT_CALL(*service_, SetActivationState(kActivationStateActivating)) |
| .Times(1); |
| EXPECT_CALL(*service_, activation_state()) |
| .Times(2) |
| .WillRepeatedly(ReturnRef(kActivationStateUnknown)); |
| capability_->UpdatePendingActivationState(); |
| VerifyAndSetActivationExpectations(); |
| |
| // Not registered. |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_SEARCHING; |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .Times(2) |
| .WillRepeatedly(Return(PendingActivationStore::kStateActivated)); |
| EXPECT_CALL(*service_, AutoConnect()).Times(0); |
| capability_->UpdatePendingActivationState(); |
| Mock::VerifyAndClearExpectations(service_); |
| |
| // Service, registered. |
| capability_->registration_state_ = MM_MODEM_3GPP_REGISTRATION_STATE_HOME; |
| EXPECT_CALL(*service_, AutoConnect()).Times(1); |
| EXPECT_CALL(*service_, activation_state()) |
| .WillOnce(ReturnRef(kActivationStateUnknown)); |
| capability_->UpdatePendingActivationState(); |
| |
| service_->set_activation_state_for_testing(kActivationStateNotActivated); |
| |
| Mock::VerifyAndClearExpectations(service_); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_CALL(*service_, activation_state()) |
| .WillRepeatedly(ReturnRef(kActivationStateUnknown)); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .WillRepeatedly(Return(PendingActivationStore::kStateUnknown)); |
| |
| // Device is connected. |
| cellular_->set_state_for_testing(Cellular::State::kConnected); |
| capability_->UpdatePendingActivationState(); |
| |
| // Device is linked. |
| cellular_->set_state_for_testing(Cellular::State::kLinked); |
| capability_->UpdatePendingActivationState(); |
| |
| // Got valid MDN, subscription_state_ is SubscriptionState::kUnknown |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| RemoveEntry(PendingActivationStore::kIdentifierICCID, kIccid)); |
| |
| cellular_->set_state_for_testing(Cellular::State::kRegistered); |
| cellular_->SetMdn("1020304"); |
| capability_->subscription_state_ = SubscriptionState::kUnknown; |
| capability_->UpdatePendingActivationState(); |
| VerifyAndSetActivationExpectations(); |
| |
| EXPECT_CALL(*service_, activation_state()) |
| .WillRepeatedly(ReturnRef(kActivationStateUnknown)); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .WillRepeatedly(Return(PendingActivationStore::kStateUnknown)); |
| |
| // Got invalid MDN, subscription_state_ is SubscriptionState::kProvisioned |
| EXPECT_CALL(*modem_info_.mock_pending_activation_store(), |
| RemoveEntry(PendingActivationStore::kIdentifierICCID, kIccid)); |
| |
| cellular_->set_state_for_testing(Cellular::State::kRegistered); |
| cellular_->SetMdn("0000000"); |
| capability_->subscription_state_ = SubscriptionState::kProvisioned; |
| capability_->UpdatePendingActivationState(); |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| TEST_F(CellularCapability3gppTest, IsServiceActivationRequired) { |
| const std::vector<MobileOperatorInfo::OnlinePortal> empty_list; |
| const std::vector<MobileOperatorInfo::OnlinePortal> olp_list{ |
| {"some@url", "some_method", "some_post_data"}}; |
| |
| capability_->subscription_state_ = SubscriptionState::kProvisioned; |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| |
| capability_->subscription_state_ = SubscriptionState::kUnprovisioned; |
| EXPECT_TRUE(capability_->IsServiceActivationRequired()); |
| |
| capability_->subscription_state_ = SubscriptionState::kUnknown; |
| cellular_->SetMdn("0000000000"); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(false)); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| Mock::VerifyAndClearExpectations(mock_home_provider_info_); |
| |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_home_provider_info_, olp_list()) |
| .WillRepeatedly(ReturnRef(empty_list)); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| Mock::VerifyAndClearExpectations(mock_home_provider_info_); |
| |
| // Set expectations for all subsequent cases. |
| EXPECT_CALL(*mock_home_provider_info_, IsMobileNetworkOperatorKnown()) |
| .WillRepeatedly(Return(true)); |
| EXPECT_CALL(*mock_home_provider_info_, olp_list()) |
| .WillRepeatedly(ReturnRef(olp_list)); |
| |
| cellular_->SetMdn(""); |
| EXPECT_TRUE(capability_->IsServiceActivationRequired()); |
| cellular_->SetMdn("1234567890"); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| cellular_->SetMdn("0000000000"); |
| EXPECT_TRUE(capability_->IsServiceActivationRequired()); |
| |
| SetDefaultCellularSimProperties(); |
| EXPECT_CALL( |
| *modem_info_.mock_pending_activation_store(), |
| GetActivationState(PendingActivationStore::kIdentifierICCID, kIccid)) |
| .WillOnce(Return(PendingActivationStore::kStateActivated)) |
| .WillOnce(Return(PendingActivationStore::kStatePending)) |
| .WillOnce(Return(PendingActivationStore::kStateUnknown)); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| EXPECT_FALSE(capability_->IsServiceActivationRequired()); |
| EXPECT_TRUE(capability_->IsServiceActivationRequired()); |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| TEST_F(CellularCapability3gppTest, OnModemCurrentCapabilitiesChanged) { |
| EXPECT_FALSE(cellular_->scanning_supported()); |
| capability_->OnModemCurrentCapabilitiesChanged(MM_MODEM_CAPABILITY_LTE); |
| EXPECT_FALSE(cellular_->scanning_supported()); |
| capability_->OnModemCurrentCapabilitiesChanged(MM_MODEM_CAPABILITY_CDMA_EVDO); |
| EXPECT_FALSE(cellular_->scanning_supported()); |
| capability_->OnModemCurrentCapabilitiesChanged(MM_MODEM_CAPABILITY_GSM_UMTS); |
| EXPECT_TRUE(cellular_->scanning_supported()); |
| capability_->OnModemCurrentCapabilitiesChanged(MM_MODEM_CAPABILITY_GSM_UMTS | |
| MM_MODEM_CAPABILITY_CDMA_EVDO); |
| EXPECT_TRUE(cellular_->scanning_supported()); |
| } |
| |
| TEST_F(CellularCapability3gppTest, SimLockStatusToProperty) { |
| Error error; |
| KeyValueStore store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_FALSE(store.Get<bool>(kSIMLockEnabledProperty)); |
| EXPECT_TRUE(store.Get<std::string>(kSIMLockTypeProperty).empty()); |
| EXPECT_EQ(0, store.Get<int32_t>(kSIMLockRetriesLeftProperty)); |
| |
| capability_->sim_lock_status_.enabled = true; |
| capability_->sim_lock_status_.retries_left = 3; |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_TRUE(store.Get<bool>(kSIMLockEnabledProperty)); |
| EXPECT_EQ("sim-pin", store.Get<std::string>(kSIMLockTypeProperty)); |
| EXPECT_EQ(3, store.Get<int32_t>(kSIMLockRetriesLeftProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PUK; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("sim-puk", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PIN2; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_TRUE(store.Get<std::string>(kSIMLockTypeProperty).empty()); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PUK2; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("sim-puk2", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_SP_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("service-provider-pin", |
| store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_SP_PUK; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("service-provider-puk", |
| store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_NET_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("network-pin", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_NET_PUK; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("network-puk", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_SIM_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("dedicated-sim", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_CORP_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("corporate-pin", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_CORP_PUK; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("corporate-puk", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_NETSUB_PIN; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("network-subset-pin", store.Get<std::string>(kSIMLockTypeProperty)); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_PH_NETSUB_PUK; |
| store = capability_->SimLockStatusToProperty(&error); |
| EXPECT_EQ("network-subset-puk", store.Get<std::string>(kSIMLockTypeProperty)); |
| } |
| |
| TEST_F(CellularCapability3gppTest, OnLockRetriesChanged) { |
| CellularCapability3gpp::LockRetryData data; |
| |
| capability_->OnLockRetriesChanged(data); |
| EXPECT_EQ(CellularCapability3gpp::kUnknownLockRetriesLeft, |
| capability_->sim_lock_status_.retries_left); |
| |
| data[MM_MODEM_LOCK_SIM_PIN] = 3; |
| data[MM_MODEM_LOCK_SIM_PIN2] = 5; |
| data[MM_MODEM_LOCK_SIM_PUK] = 10; |
| capability_->OnLockRetriesChanged(data); |
| EXPECT_EQ(3, capability_->sim_lock_status_.retries_left); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PUK; |
| capability_->OnLockRetriesChanged(data); |
| EXPECT_EQ(10, capability_->sim_lock_status_.retries_left); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PIN; |
| capability_->OnLockRetriesChanged(data); |
| EXPECT_EQ(3, capability_->sim_lock_status_.retries_left); |
| |
| capability_->sim_lock_status_.lock_type = MM_MODEM_LOCK_SIM_PIN2; |
| capability_->OnLockRetriesChanged(data); |
| // retries_left should indicate the number of SIM_PIN retries if the |
| // lock is SIM_PIN or SIM_PIN2 |
| EXPECT_EQ(3, capability_->sim_lock_status_.retries_left); |
| |
| data.clear(); |
| capability_->OnLockRetriesChanged(data); |
| EXPECT_EQ(CellularCapability3gpp::kUnknownLockRetriesLeft, |
| capability_->sim_lock_status_.retries_left); |
| } |
| |
| TEST_F(CellularCapability3gppTest, OnLockTypeChanged) { |
| EXPECT_EQ(MM_MODEM_LOCK_UNKNOWN, capability_->sim_lock_status_.lock_type); |
| |
| capability_->OnLockTypeChanged(MM_MODEM_LOCK_NONE); |
| EXPECT_EQ(MM_MODEM_LOCK_NONE, capability_->sim_lock_status_.lock_type); |
| EXPECT_FALSE(capability_->sim_lock_status_.enabled); |
| |
| capability_->OnLockTypeChanged(MM_MODEM_LOCK_SIM_PIN); |
| EXPECT_EQ(MM_MODEM_LOCK_SIM_PIN, capability_->sim_lock_status_.lock_type); |
| EXPECT_TRUE(capability_->sim_lock_status_.enabled); |
| |
| capability_->sim_lock_status_.enabled = false; |
| capability_->OnLockTypeChanged(MM_MODEM_LOCK_SIM_PUK); |
| EXPECT_EQ(MM_MODEM_LOCK_SIM_PUK, capability_->sim_lock_status_.lock_type); |
| EXPECT_TRUE(capability_->sim_lock_status_.enabled); |
| } |
| |
| TEST_F(CellularCapability3gppTest, OnSimLockPropertiesChanged) { |
| EXPECT_EQ(MM_MODEM_LOCK_UNKNOWN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(0, capability_->sim_lock_status_.retries_left); |
| |
| KeyValueStore changed; |
| |
| capability_->OnModemPropertiesChanged(changed); |
| EXPECT_EQ(MM_MODEM_LOCK_UNKNOWN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(0, capability_->sim_lock_status_.retries_left); |
| |
| // Unlock retries changed, but the SIM wasn't locked. |
| CellularCapability3gpp::LockRetryData retry_data; |
| retry_data[MM_MODEM_LOCK_SIM_PIN] = 3; |
| changed.SetVariant(MM_MODEM_PROPERTY_UNLOCKRETRIES, brillo::Any(retry_data)); |
| |
| capability_->OnModemPropertiesChanged(changed); |
| EXPECT_EQ(MM_MODEM_LOCK_UNKNOWN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(3, capability_->sim_lock_status_.retries_left); |
| |
| // Unlock retries changed and the SIM got locked. |
| changed.Set<uint32_t>(MM_MODEM_PROPERTY_UNLOCKREQUIRED, |
| static_cast<uint32_t>(MM_MODEM_LOCK_SIM_PIN)); |
| capability_->OnModemPropertiesChanged(changed); |
| EXPECT_EQ(MM_MODEM_LOCK_SIM_PIN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(3, capability_->sim_lock_status_.retries_left); |
| |
| // Only unlock retries changed. |
| changed.Remove(MM_MODEM_PROPERTY_UNLOCKREQUIRED); |
| retry_data[MM_MODEM_LOCK_SIM_PIN] = 2; |
| changed.SetVariant(MM_MODEM_PROPERTY_UNLOCKRETRIES, brillo::Any(retry_data)); |
| capability_->OnModemPropertiesChanged(changed); |
| EXPECT_EQ(MM_MODEM_LOCK_SIM_PIN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(2, capability_->sim_lock_status_.retries_left); |
| |
| // Unlock retries changed with a value that doesn't match the current |
| // lock type. Default to unknown if PIN1 is unavailable. |
| retry_data.clear(); |
| retry_data[MM_MODEM_LOCK_SIM_PIN2] = 2; |
| changed.SetVariant(MM_MODEM_PROPERTY_UNLOCKRETRIES, brillo::Any(retry_data)); |
| capability_->OnModemPropertiesChanged(changed); |
| EXPECT_EQ(MM_MODEM_LOCK_SIM_PIN, capability_->sim_lock_status_.lock_type); |
| EXPECT_EQ(CellularCapability3gpp::kUnknownLockRetriesLeft, |
| capability_->sim_lock_status_.retries_left); |
| } |
| |
| TEST_F(CellularCapability3gppTest, MultiSimProperties) { |
| InitProxies(); |
| |
| const char kIccid1[] = "110100000001"; |
| const char kEid1[] = "110100000002"; |
| KeyValueStore sim_properties1; |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid1); |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_EID, kEid1); |
| SetSimProperties(kSimPath1, sim_properties1); |
| |
| const char kIccid2[] = "210100000001"; |
| const char kEid2[] = "210100000002"; |
| KeyValueStore sim_properties2; |
| sim_properties2.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid2); |
| sim_properties2.Set<std::string>(MM_SIM_PROPERTY_EID, kEid2); |
| SetSimProperties(kSimPath2, sim_properties2); |
| |
| UpdateSims(kSimPath1); |
| |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_EQ(kIccid1, cellular_->iccid()); |
| EXPECT_EQ(kEid1, cellular_->eid()); |
| |
| const KeyValueStores& sim_slot_info = cellular_->sim_slot_info_for_testing(); |
| ASSERT_EQ(2u, sim_slot_info.size()); |
| EXPECT_EQ(sim_slot_info[0].Get<std::string>(kSIMSlotInfoICCID), kIccid1); |
| EXPECT_EQ(sim_slot_info[0].Get<std::string>(kSIMSlotInfoEID), kEid1); |
| EXPECT_EQ(sim_slot_info[1].Get<std::string>(kSIMSlotInfoICCID), kIccid2); |
| EXPECT_EQ(sim_slot_info[1].Get<std::string>(kSIMSlotInfoEID), kEid2); |
| VerifyAndSetActivationExpectations(); |
| |
| // Switch active slot to 2. |
| KeyValueStore modem_properties; |
| modem_properties.Set<RpcIdentifier>(MM_MODEM_PROPERTY_SIM, kSimPath2); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| dispatcher_.DispatchPendingEvents(); |
| |
| EXPECT_EQ(kSimPath2, capability_->sim_path_for_testing()); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_EQ(kIccid2, cellular_->iccid()); |
| EXPECT_EQ(kEid2, cellular_->eid()); |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| // Test behavior when a SIM path is set but not SIMSLOTS. |
| TEST_F(CellularCapability3gppTest, SimPathOnly) { |
| InitProxies(); |
| |
| const char kIccid1[] = "110100000001"; |
| const char kEid1[] = "110100000002"; |
| KeyValueStore sim_properties; |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid1); |
| sim_properties.Set<std::string>(MM_SIM_PROPERTY_EID, kEid1); |
| SetSimProperties(kSimPath1, sim_properties); |
| |
| KeyValueStore modem_properties; |
| modem_properties.Set<RpcIdentifier>(MM_MODEM_PROPERTY_SIM, kSimPath1); |
| capability_->OnPropertiesChanged(MM_DBUS_INTERFACE_MODEM, modem_properties); |
| dispatcher_.DispatchPendingEvents(); |
| |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_EQ(kIccid1, cellular_->iccid()); |
| EXPECT_EQ(kEid1, cellular_->eid()); |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| // Tests that when the primary SIM ICCID is empty, and another SIM has a valid |
| // ICCID, the other SIM is selected. |
| TEST_F(CellularCapability3gppTest, SetPrimarySimSlot) { |
| EXPECT_CALL(*modem_proxy_, SetPrimarySimSlot(2, _, _)).Times(1); |
| InitProxies(); |
| |
| const char kIccid1[] = ""; |
| const char kEid1[] = "110100000002"; |
| KeyValueStore sim_properties1; |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid1); |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_EID, kEid1); |
| SetSimProperties(kSimPath1, sim_properties1); |
| |
| const char kIccid2[] = "210100000001"; |
| const char kEid2[] = "210100000002"; |
| KeyValueStore sim_properties2; |
| sim_properties2.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid2); |
| sim_properties2.Set<std::string>(MM_SIM_PROPERTY_EID, kEid2); |
| SetSimProperties(kSimPath2, sim_properties2); |
| |
| UpdateSims(kSimPath1); |
| |
| // TODO(b/169581681): Fake Modem.SetPrimarySimSlot() behavior to provide |
| // updated SIM properties. |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| TEST_F(CellularCapability3gppTest, EmptySimSlot) { |
| InitProxies(); |
| |
| const char kIccid1[] = "110100000001"; |
| const char kEid1[] = "110100000002"; |
| KeyValueStore sim_properties1; |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_SIMIDENTIFIER, kIccid1); |
| sim_properties1.Set<std::string>(MM_SIM_PROPERTY_EID, kEid1); |
| SetSimProperties(kSimPath1, sim_properties1); |
| |
| KeyValueStore sim_properties2; |
| SetSimProperties(CellularCapability3gpp::kRootPath, sim_properties2); |
| |
| UpdateSims(kSimPath1); |
| |
| EXPECT_EQ(kSimPath1, capability_->sim_path_for_testing()); |
| EXPECT_TRUE(cellular_->sim_present()); |
| EXPECT_EQ(kIccid1, cellular_->iccid()); |
| EXPECT_EQ(kEid1, cellular_->eid()); |
| |
| const KeyValueStores& sim_slot_info = cellular_->sim_slot_info_for_testing(); |
| ASSERT_EQ(2u, sim_slot_info.size()); |
| EXPECT_EQ(sim_slot_info[0].Get<std::string>(kSIMSlotInfoICCID), kIccid1); |
| EXPECT_EQ(sim_slot_info[0].Get<std::string>(kSIMSlotInfoEID), kEid1); |
| EXPECT_TRUE(sim_slot_info[1].Get<std::string>(kSIMSlotInfoICCID).empty()); |
| EXPECT_TRUE(sim_slot_info[1].Get<std::string>(kSIMSlotInfoEID).empty()); |
| VerifyAndSetActivationExpectations(); |
| } |
| |
| } // namespace shill |