// // Copyright (C) 2012 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #include "shill/cellular/cellular_capability_gsm.h" #include #include #include #include #if defined(__ANDROID__) #include #else #include #endif // __ANDROID__ #include #include "shill/cellular/cellular.h" #include "shill/cellular/cellular_service.h" #include "shill/cellular/mock_mobile_operator_info.h" #include "shill/cellular/mock_modem_gsm_card_proxy.h" #include "shill/cellular/mock_modem_gsm_network_proxy.h" #include "shill/cellular/mock_modem_info.h" #include "shill/cellular/mock_modem_proxy.h" #include "shill/cellular/mock_modem_simple_proxy.h" #include "shill/error.h" #include "shill/mock_adaptors.h" #include "shill/mock_control.h" #include "shill/mock_log.h" #include "shill/mock_profile.h" #include "shill/test_event_dispatcher.h" #include "shill/testing.h" using base::Bind; using base::StringPrintf; using base::Unretained; using std::string; using std::vector; using testing::_; using testing::Invoke; using testing::NiceMock; using testing::Return; using testing::SaveArg; namespace shill { class CellularCapabilityGSMTest : public testing::Test { public: CellularCapabilityGSMTest() : control_interface_(this), modem_info_(&control_interface_, &dispatcher_, nullptr, nullptr), create_card_proxy_from_factory_(false), proxy_(new MockModemProxy()), simple_proxy_(new MockModemSimpleProxy()), card_proxy_(new MockModemGSMCardProxy()), network_proxy_(new MockModemGSMNetworkProxy()), capability_(nullptr), device_adaptor_(nullptr), cellular_(new Cellular(&modem_info_, "", kAddress, 0, Cellular::kTypeGSM, "", "")), mock_home_provider_info_(nullptr), mock_serving_operator_info_(nullptr) { modem_info_.metrics()->RegisterDevice(cellular_->interface_index(), Technology::kCellular); } virtual ~CellularCapabilityGSMTest() { cellular_->service_ = nullptr; capability_ = nullptr; device_adaptor_ = nullptr; } virtual void SetUp() { capability_ = static_cast(cellular_->capability_.get()); device_adaptor_ = static_cast(cellular_->adaptor()); } void InvokeEnable(bool enable, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeGetIMEI(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run(kIMEI, Error()); } void InvokeGetIMSI(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run(kIMSI, Error()); } void InvokeGetIMSIFails(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run("", Error(Error::kOperationFailed)); } void InvokeGetMSISDN(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run(kMSISDN, Error()); } void InvokeGetMSISDNFail(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run("", Error(Error::kOperationFailed)); } void InvokeGetSPN(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run(kTestCarrier, Error()); } void InvokeGetSPNFail(Error* error, const GSMIdentifierCallback& callback, int timeout) { callback.Run("", Error(Error::kOperationFailed)); } void InvokeGetSignalQuality(Error* error, const SignalQualityCallback& callback, int timeout) { callback.Run(kStrength, Error()); } void InvokeGetRegistrationInfo(Error* error, const RegistrationInfoCallback& callback, int timeout) { callback.Run(MM_MODEM_GSM_NETWORK_REG_STATUS_HOME, kTestNetwork, kTestCarrier, Error()); } void InvokeRegister(const string& network_id, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeEnablePIN(const string& pin, bool enable, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeSendPIN(const string& pin, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeSendPUK(const string& puk, const string& pin, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeChangePIN(const string& old_pin, const string& pin, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error()); } void InvokeGetModemStatus(Error* error, const KeyValueStoreCallback& callback, int timeout) { KeyValueStore props; callback.Run(props, Error()); } void InvokeGetModemInfo(Error* error, const ModemInfoCallback& callback, int timeout) { callback.Run("", "", "", Error()); } void InvokeConnectFail(KeyValueStore props, Error* error, const ResultCallback& callback, int timeout) { callback.Run(Error(Error::kOperationFailed)); } MOCK_METHOD1(TestCallback, void(const Error& error)); protected: static const char kAddress[]; static const char kTestMobileProviderDBPath[]; static const char kTestNetwork[]; static const char kTestCarrier[]; static const char kPIN[]; static const char kPUK[]; static const char kIMEI[]; static const char kIMSI[]; static const char kMSISDN[]; static const int kStrength; class TestControl : public MockControl { public: explicit TestControl(CellularCapabilityGSMTest* test) : test_(test) {} virtual ModemProxyInterface* CreateModemProxy( const string& /*path*/, const string& /*service*/) { return test_->proxy_.release(); } virtual ModemSimpleProxyInterface* CreateModemSimpleProxy( const string& /*path*/, const string& /*service*/) { return test_->simple_proxy_.release(); } virtual ModemGSMCardProxyInterface* CreateModemGSMCardProxy( const string& /*path*/, const string& /*service*/) { // TODO(benchan): This code conditionally returns a nullptr to avoid // CellularCapabilityGSM::InitProperties (and thus // CellularCapabilityGSM::GetIMSI) from being called during the // construction. Remove this workaround after refactoring the tests. return test_->create_card_proxy_from_factory_ ? test_->card_proxy_.release() : nullptr; } virtual ModemGSMNetworkProxyInterface* CreateModemGSMNetworkProxy( const string& /*path*/, const string& /*service*/) { return test_->network_proxy_.release(); } private: CellularCapabilityGSMTest* test_; }; void SetProxy() { capability_->proxy_.reset(proxy_.release()); } void SetCardProxy() { capability_->card_proxy_.reset(card_proxy_.release()); } void SetNetworkProxy() { capability_->network_proxy_.reset(network_proxy_.release()); } void SetAccessTechnology(uint32_t technology) { capability_->access_technology_ = technology; } void SetRegistrationState(uint32_t state) { capability_->registration_state_ = state; } void CreateService() { // The following constants are never directly accessed by the tests. const char kStorageIdentifier[] = "default_test_storage_id"; 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(&modem_info_, cellular_); service->SetStorageIdentifier(kStorageIdentifier); service->SetFriendlyName(kFriendlyServiceName); Stringmap serving_operator; serving_operator[kOperatorCodeKey] = kOperatorCode; serving_operator[kOperatorNameKey] = kOperatorName; serving_operator[kOperatorCountryKey] = kOperatorCountry; service->set_serving_operator(serving_operator); cellular_->set_home_provider(serving_operator); cellular_->service_ = service; } void SetMockMobileOperatorInfoObjects() { CHECK(!mock_home_provider_info_); CHECK(!mock_serving_operator_info_); mock_home_provider_info_ = new MockMobileOperatorInfo(&dispatcher_, "HomeProvider"); mock_serving_operator_info_ = new MockMobileOperatorInfo(&dispatcher_, "ServingOperator"); cellular_->set_home_provider_info(mock_home_provider_info_); cellular_->set_serving_operator_info(mock_serving_operator_info_); } void SetupCommonProxiesExpectations() { EXPECT_CALL(*proxy_, set_state_changed_callback(_)); EXPECT_CALL(*network_proxy_, set_signal_quality_callback(_)); EXPECT_CALL(*network_proxy_, set_network_mode_callback(_)); EXPECT_CALL(*network_proxy_, set_registration_info_callback(_)); } void SetupCommonStartModemExpectations() { SetupCommonProxiesExpectations(); EXPECT_CALL(*proxy_, Enable(_, _, _, CellularCapability::kTimeoutEnable)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeEnable)); EXPECT_CALL(*card_proxy_, GetIMEI(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetIMEI)); EXPECT_CALL(*card_proxy_, GetIMSI(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetIMSI)); EXPECT_CALL(*network_proxy_, AccessTechnology()); EXPECT_CALL(*card_proxy_, EnabledFacilityLocks()); EXPECT_CALL(*proxy_, GetModemInfo(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetModemInfo)); EXPECT_CALL(*network_proxy_, GetRegistrationInfo(_, _, CellularCapability::kTimeoutDefault)); EXPECT_CALL(*network_proxy_, GetSignalQuality(_, _, CellularCapability::kTimeoutDefault)); EXPECT_CALL(*this, TestCallback(IsSuccess())); } void InitProxies() { AllowCreateCardProxyFromFactory(); capability_->InitProxies(); } void AllowCreateCardProxyFromFactory() { create_card_proxy_from_factory_ = true; } EventDispatcherForTest dispatcher_; TestControl control_interface_; MockModemInfo modem_info_; bool create_card_proxy_from_factory_; std::unique_ptr proxy_; std::unique_ptr simple_proxy_; std::unique_ptr card_proxy_; std::unique_ptr network_proxy_; CellularCapabilityGSM* capability_; // Owned by |cellular_|. DeviceMockAdaptor* device_adaptor_; // Owned by |cellular_|. CellularRefPtr cellular_; // Set when required and passed to |cellular_|. Owned by |cellular_|. MockMobileOperatorInfo* mock_home_provider_info_; MockMobileOperatorInfo* mock_serving_operator_info_; }; const char CellularCapabilityGSMTest::kAddress[] = "1122334455"; const char CellularCapabilityGSMTest::kTestMobileProviderDBPath[] = "provider_db_unittest.bfd"; const char CellularCapabilityGSMTest::kTestCarrier[] = "The Cellular Carrier"; const char CellularCapabilityGSMTest::kTestNetwork[] = "310555"; const char CellularCapabilityGSMTest::kPIN[] = "9876"; const char CellularCapabilityGSMTest::kPUK[] = "8765"; const char CellularCapabilityGSMTest::kIMEI[] = "987654321098765"; const char CellularCapabilityGSMTest::kIMSI[] = "310150123456789"; const char CellularCapabilityGSMTest::kMSISDN[] = "12345678901"; const int CellularCapabilityGSMTest::kStrength = 80; TEST_F(CellularCapabilityGSMTest, PropertyStore) { EXPECT_TRUE(cellular_->store().Contains(kSIMLockStatusProperty)); } TEST_F(CellularCapabilityGSMTest, GetIMEI) { EXPECT_CALL(*card_proxy_, GetIMEI(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetIMEI)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); ASSERT_TRUE(cellular_->imei().empty()); capability_->GetIMEI(Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_EQ(kIMEI, cellular_->imei()); } TEST_F(CellularCapabilityGSMTest, GetIMSI) { SetMockMobileOperatorInfoObjects(); EXPECT_CALL(*card_proxy_, GetIMSI(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetIMSI)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); ResultCallback callback = Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this)); EXPECT_TRUE(cellular_->imsi().empty()); EXPECT_FALSE(cellular_->sim_present()); EXPECT_CALL(*mock_home_provider_info_, UpdateIMSI(kIMSI)); capability_->GetIMSI(callback); EXPECT_EQ(kIMSI, cellular_->imsi()); EXPECT_TRUE(cellular_->sim_present()); } // In this test, the call to the proxy's GetIMSI() will always indicate failure, // which will cause the retry logic to call the proxy again a number of times. // Eventually, the retries expire. TEST_F(CellularCapabilityGSMTest, GetIMSIFails) { ScopedMockLog log; EXPECT_CALL(log, Log(logging::LOG_INFO, ::testing::EndsWith("cellular_capability_gsm.cc"), ::testing::StartsWith("GetIMSI failed - "))); EXPECT_CALL(*card_proxy_, GetIMSI(_, _, CellularCapability::kTimeoutDefault)) .Times(CellularCapabilityGSM::kGetIMSIRetryLimit + 2) .WillRepeatedly(Invoke(this, &CellularCapabilityGSMTest::InvokeGetIMSIFails)); EXPECT_CALL(*this, TestCallback(IsFailure())).Times(2); SetCardProxy(); ResultCallback callback = Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this)); EXPECT_TRUE(cellular_->imsi().empty()); EXPECT_FALSE(cellular_->sim_present()); capability_->sim_lock_status_.lock_type = "sim-pin"; capability_->GetIMSI(callback); EXPECT_TRUE(cellular_->imsi().empty()); EXPECT_TRUE(cellular_->sim_present()); capability_->sim_lock_status_.lock_type.clear(); cellular_->set_sim_present(false); capability_->get_imsi_retries_ = 0; EXPECT_EQ(CellularCapabilityGSM::kGetIMSIRetryDelayMilliseconds, capability_->get_imsi_retry_delay_milliseconds_); // Set the delay to zero to speed up the test. capability_->get_imsi_retry_delay_milliseconds_ = 0; capability_->GetIMSI(callback); for (int i = 0; i < CellularCapabilityGSM::kGetIMSIRetryLimit; ++i) { dispatcher_.DispatchPendingEvents(); } EXPECT_EQ(CellularCapabilityGSM::kGetIMSIRetryLimit + 1, capability_->get_imsi_retries_); EXPECT_TRUE(cellular_->imsi().empty()); EXPECT_FALSE(cellular_->sim_present()); } TEST_F(CellularCapabilityGSMTest, GetMSISDN) { EXPECT_CALL(*card_proxy_, GetMSISDN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetMSISDN)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); ASSERT_TRUE(cellular_->mdn().empty()); capability_->GetMSISDN(Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_EQ(kMSISDN, cellular_->mdn()); } TEST_F(CellularCapabilityGSMTest, GetSPN) { EXPECT_CALL(*card_proxy_, GetSPN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetSPN)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); ASSERT_TRUE(capability_->spn_.empty()); capability_->GetSPN(Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_EQ(kTestCarrier, capability_->spn_); } TEST_F(CellularCapabilityGSMTest, GetSignalQuality) { EXPECT_CALL(*network_proxy_, GetSignalQuality(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetSignalQuality)); SetNetworkProxy(); CreateService(); EXPECT_EQ(0, cellular_->service()->strength()); capability_->GetSignalQuality(); EXPECT_EQ(kStrength, cellular_->service()->strength()); } TEST_F(CellularCapabilityGSMTest, RegisterOnNetwork) { EXPECT_CALL(*network_proxy_, Register(kTestNetwork, _, _, CellularCapability::kTimeoutRegister)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeRegister)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetNetworkProxy(); Error error; capability_->RegisterOnNetwork(kTestNetwork, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_EQ(kTestNetwork, cellular_->selected_network()); } TEST_F(CellularCapabilityGSMTest, IsRegistered) { EXPECT_FALSE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_IDLE); EXPECT_FALSE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_HOME); EXPECT_TRUE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_SEARCHING); EXPECT_FALSE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_DENIED); EXPECT_FALSE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_UNKNOWN); EXPECT_FALSE(capability_->IsRegistered()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_ROAMING); EXPECT_TRUE(capability_->IsRegistered()); } TEST_F(CellularCapabilityGSMTest, GetRegistrationState) { ASSERT_FALSE(capability_->IsRegistered()); EXPECT_CALL(*network_proxy_, GetRegistrationInfo(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetRegistrationInfo)); SetNetworkProxy(); capability_->GetRegistrationState(); EXPECT_TRUE(capability_->IsRegistered()); EXPECT_EQ(MM_MODEM_GSM_NETWORK_REG_STATUS_HOME, capability_->registration_state_); } TEST_F(CellularCapabilityGSMTest, RequirePIN) { EXPECT_CALL(*card_proxy_, EnablePIN(kPIN, true, _, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeEnablePIN)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); Error error; capability_->RequirePIN(kPIN, true, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_TRUE(error.IsSuccess()); } TEST_F(CellularCapabilityGSMTest, EnterPIN) { EXPECT_CALL(*card_proxy_, SendPIN(kPIN, _, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeSendPIN)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); Error error; capability_->EnterPIN(kPIN, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_TRUE(error.IsSuccess()); } TEST_F(CellularCapabilityGSMTest, UnblockPIN) { EXPECT_CALL(*card_proxy_, SendPUK(kPUK, kPIN, _, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeSendPUK)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); Error error; capability_->UnblockPIN(kPUK, kPIN, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_TRUE(error.IsSuccess()); } TEST_F(CellularCapabilityGSMTest, ChangePIN) { static const char kOldPIN[] = "1111"; EXPECT_CALL(*card_proxy_, ChangePIN(kOldPIN, kPIN, _, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeChangePIN)); EXPECT_CALL(*this, TestCallback(IsSuccess())); SetCardProxy(); Error error; capability_->ChangePIN(kOldPIN, kPIN, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); EXPECT_TRUE(error.IsSuccess()); } TEST_F(CellularCapabilityGSMTest, ParseScanResult) { static const char kID[] = "123"; static const char kLongName[] = "long name"; static const char kShortName[] = "short name"; GSMScanResult result; result[CellularCapabilityGSM::kNetworkPropertyStatus] = "1"; result[CellularCapabilityGSM::kNetworkPropertyID] = kID; result[CellularCapabilityGSM::kNetworkPropertyLongName] = kLongName; result[CellularCapabilityGSM::kNetworkPropertyShortName] = kShortName; result[CellularCapabilityGSM::kNetworkPropertyAccessTechnology] = "3"; result["unknown property"] = "random value"; Stringmap parsed = capability_->ParseScanResult(result); EXPECT_EQ(5, parsed.size()); EXPECT_EQ("available", parsed[kStatusProperty]); EXPECT_EQ(kID, parsed[kNetworkIdProperty]); EXPECT_EQ(kLongName, parsed[kLongNameProperty]); EXPECT_EQ(kShortName, parsed[kShortNameProperty]); EXPECT_EQ(kNetworkTechnologyEdge, parsed[kTechnologyProperty]); } TEST_F(CellularCapabilityGSMTest, ParseScanResultProviderLookup) { static const char kID[] = "10001"; const string kLongName = "TestNetworkLongName"; // Replace the |MobileOperatorInfo| used by |ParseScanResult| by a mock. auto* mock_mobile_operator_info = new MockMobileOperatorInfo( &dispatcher_, "MockParseScanResult"); capability_->mobile_operator_info_.reset(mock_mobile_operator_info); mock_mobile_operator_info->SetEmptyDefaultsForProperties(); EXPECT_CALL(*mock_mobile_operator_info, UpdateMCCMNC(kID)); EXPECT_CALL(*mock_mobile_operator_info, IsMobileNetworkOperatorKnown()). WillOnce(Return(true)); EXPECT_CALL(*mock_mobile_operator_info, operator_name()). WillRepeatedly(ReturnRef(kLongName)); GSMScanResult result; result[CellularCapabilityGSM::kNetworkPropertyID] = kID; Stringmap parsed = capability_->ParseScanResult(result); EXPECT_EQ(2, parsed.size()); EXPECT_EQ(kID, parsed[kNetworkIdProperty]); EXPECT_EQ(kLongName, parsed[kLongNameProperty]); } TEST_F(CellularCapabilityGSMTest, SetAccessTechnology) { capability_->SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_GSM); EXPECT_EQ(MM_MODEM_GSM_ACCESS_TECH_GSM, capability_->access_technology_); CreateService(); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_HOME); capability_->SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_GPRS); EXPECT_EQ(MM_MODEM_GSM_ACCESS_TECH_GPRS, capability_->access_technology_); EXPECT_EQ(kNetworkTechnologyGprs, cellular_->service()->network_technology()); } TEST_F(CellularCapabilityGSMTest, AllowRoaming) { EXPECT_FALSE(cellular_->allow_roaming_); EXPECT_FALSE(cellular_->provider_requires_roaming()); EXPECT_FALSE(capability_->AllowRoaming()); cellular_->set_provider_requires_roaming(true); EXPECT_TRUE(capability_->AllowRoaming()); cellular_->set_provider_requires_roaming(false); cellular_->allow_roaming_ = true; EXPECT_TRUE(capability_->AllowRoaming()); } TEST_F(CellularCapabilityGSMTest, GetNetworkTechnologyString) { EXPECT_EQ("", capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_GSM); EXPECT_EQ(kNetworkTechnologyGsm, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_GSM_COMPACT); EXPECT_EQ(kNetworkTechnologyGsm, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_GPRS); EXPECT_EQ(kNetworkTechnologyGprs, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_EDGE); EXPECT_EQ(kNetworkTechnologyEdge, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_UMTS); EXPECT_EQ(kNetworkTechnologyUmts, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_HSDPA); EXPECT_EQ(kNetworkTechnologyHspa, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_HSUPA); EXPECT_EQ(kNetworkTechnologyHspa, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_HSPA); EXPECT_EQ(kNetworkTechnologyHspa, capability_->GetNetworkTechnologyString()); SetAccessTechnology(MM_MODEM_GSM_ACCESS_TECH_HSPA_PLUS); EXPECT_EQ(kNetworkTechnologyHspaPlus, capability_->GetNetworkTechnologyString()); } TEST_F(CellularCapabilityGSMTest, GetRoamingStateString) { EXPECT_EQ(kRoamingStateUnknown, capability_->GetRoamingStateString()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_HOME); EXPECT_EQ(kRoamingStateHome, capability_->GetRoamingStateString()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_ROAMING); EXPECT_EQ(kRoamingStateRoaming, capability_->GetRoamingStateString()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_SEARCHING); EXPECT_EQ(kRoamingStateUnknown, capability_->GetRoamingStateString()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_DENIED); EXPECT_EQ(kRoamingStateUnknown, capability_->GetRoamingStateString()); SetRegistrationState(MM_MODEM_GSM_NETWORK_REG_STATUS_IDLE); EXPECT_EQ(kRoamingStateUnknown, capability_->GetRoamingStateString()); } TEST_F(CellularCapabilityGSMTest, OnPropertiesChanged) { EXPECT_EQ(MM_MODEM_GSM_ACCESS_TECH_UNKNOWN, capability_->access_technology_); EXPECT_FALSE(capability_->sim_lock_status_.enabled); EXPECT_EQ("", capability_->sim_lock_status_.lock_type); EXPECT_EQ(0, capability_->sim_lock_status_.retries_left); KeyValueStore props; static const char kLockType[] = "sim-pin"; const int kRetries = 3; props.SetUint(CellularCapabilityGSM::kPropertyAccessTechnology, MM_MODEM_GSM_ACCESS_TECH_EDGE); props.SetUint(CellularCapabilityGSM::kPropertyEnabledFacilityLocks, MM_MODEM_GSM_FACILITY_SIM); props.SetString(CellularCapabilityGSM::kPropertyUnlockRequired, kLockType); props.SetUint(CellularCapabilityGSM::kPropertyUnlockRetries, kRetries); // Call with the 'wrong' interface and nothing should change. capability_->OnPropertiesChanged(MM_MODEM_GSM_INTERFACE, props, vector()); EXPECT_EQ(MM_MODEM_GSM_ACCESS_TECH_UNKNOWN, capability_->access_technology_); EXPECT_FALSE(capability_->sim_lock_status_.enabled); EXPECT_EQ("", capability_->sim_lock_status_.lock_type); EXPECT_EQ(0, capability_->sim_lock_status_.retries_left); // Call with the MM_MODEM_GSM_NETWORK_INTERFACE interface and expect a change // to the enabled state of the SIM lock. KeyValueStore lock_status; lock_status.SetBool(kSIMLockEnabledProperty, true); lock_status.SetString(kSIMLockTypeProperty, ""); lock_status.SetUint(kSIMLockRetriesLeftProperty, 0); EXPECT_CALL(*device_adaptor_, EmitKeyValueStoreChanged( kSIMLockStatusProperty, KeyValueStoreEq(lock_status))); capability_->OnPropertiesChanged(MM_MODEM_GSM_NETWORK_INTERFACE, props, vector()); EXPECT_EQ(MM_MODEM_GSM_ACCESS_TECH_EDGE, capability_->access_technology_); capability_->OnPropertiesChanged(MM_MODEM_GSM_CARD_INTERFACE, props, vector()); EXPECT_TRUE(capability_->sim_lock_status_.enabled); EXPECT_TRUE(capability_->sim_lock_status_.lock_type.empty()); EXPECT_EQ(0, capability_->sim_lock_status_.retries_left); // Some properties are sent on the MM_MODEM_INTERFACE. capability_->sim_lock_status_.enabled = false; capability_->sim_lock_status_.lock_type = ""; capability_->sim_lock_status_.retries_left = 0; KeyValueStore lock_status2; lock_status2.SetBool(kSIMLockEnabledProperty, false); lock_status2.SetString(kSIMLockTypeProperty, kLockType); lock_status2.SetUint(kSIMLockRetriesLeftProperty, kRetries); EXPECT_CALL(*device_adaptor_, EmitKeyValueStoreChanged(kSIMLockStatusProperty, KeyValueStoreEq(lock_status2))); capability_->OnPropertiesChanged(MM_MODEM_INTERFACE, props, vector()); EXPECT_FALSE(capability_->sim_lock_status_.enabled); EXPECT_EQ(kLockType, capability_->sim_lock_status_.lock_type); EXPECT_EQ(kRetries, capability_->sim_lock_status_.retries_left); } TEST_F(CellularCapabilityGSMTest, StartModemSuccess) { SetupCommonStartModemExpectations(); EXPECT_CALL(*card_proxy_, GetSPN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetSPN)); EXPECT_CALL(*card_proxy_, GetMSISDN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetMSISDN)); AllowCreateCardProxyFromFactory(); Error error; capability_->StartModem( &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); dispatcher_.DispatchPendingEvents(); } TEST_F(CellularCapabilityGSMTest, StartModemGetSPNFail) { SetupCommonStartModemExpectations(); EXPECT_CALL(*card_proxy_, GetSPN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetSPNFail)); EXPECT_CALL(*card_proxy_, GetMSISDN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetMSISDN)); AllowCreateCardProxyFromFactory(); Error error; capability_->StartModem( &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); dispatcher_.DispatchPendingEvents(); } TEST_F(CellularCapabilityGSMTest, StartModemGetMSISDNFail) { SetupCommonStartModemExpectations(); EXPECT_CALL(*card_proxy_, GetSPN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetSPN)); EXPECT_CALL(*card_proxy_, GetMSISDN(_, _, CellularCapability::kTimeoutDefault)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeGetMSISDNFail)); AllowCreateCardProxyFromFactory(); Error error; capability_->StartModem( &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); dispatcher_.DispatchPendingEvents(); } TEST_F(CellularCapabilityGSMTest, ConnectFailureNoService) { // Make sure we don't crash if the connect failed and there is no // CellularService object. This can happen if the modem is enabled and // then quickly disabled. SetupCommonProxiesExpectations(); EXPECT_CALL(*simple_proxy_, Connect(_, _, _, CellularCapabilityGSM::kTimeoutConnect)) .WillOnce(Invoke(this, &CellularCapabilityGSMTest::InvokeConnectFail)); EXPECT_CALL(*this, TestCallback(IsFailure())); InitProxies(); EXPECT_FALSE(capability_->cellular()->service()); Error error; KeyValueStore props; capability_->Connect(props, &error, Bind(&CellularCapabilityGSMTest::TestCallback, Unretained(this))); } } // namespace shill