175 lines
4.9 KiB
C++
175 lines
4.9 KiB
C++
//
|
|
// 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/modem_manager.h"
|
|
|
|
#include <base/stl_util.h>
|
|
#include <mm/mm-modem.h>
|
|
|
|
#include "shill/cellular/modem.h"
|
|
#include "shill/cellular/modem_manager_proxy_interface.h"
|
|
#include "shill/control_interface.h"
|
|
#include "shill/error.h"
|
|
#include "shill/logging.h"
|
|
#include "shill/manager.h"
|
|
|
|
using std::string;
|
|
using std::shared_ptr;
|
|
using std::vector;
|
|
|
|
namespace shill {
|
|
|
|
ModemManager::ModemManager(ControlInterface* control_interface,
|
|
const string& service,
|
|
const string& path,
|
|
ModemInfo* modem_info)
|
|
: control_interface_(control_interface),
|
|
service_(service),
|
|
path_(path),
|
|
service_connected_(false),
|
|
modem_info_(modem_info) {}
|
|
|
|
ModemManager::~ModemManager() {}
|
|
|
|
void ModemManager::Connect() {
|
|
// Inheriting classes call this superclass method.
|
|
service_connected_ = true;
|
|
}
|
|
|
|
void ModemManager::Disconnect() {
|
|
// Inheriting classes call this superclass method.
|
|
modems_.clear();
|
|
service_connected_ = false;
|
|
}
|
|
|
|
void ModemManager::OnAppeared() {
|
|
LOG(INFO) << "Modem manager " << service_ << " appeared.";
|
|
Connect();
|
|
}
|
|
|
|
void ModemManager::OnVanished() {
|
|
LOG(INFO) << "Modem manager " << service_ << " vanished.";
|
|
Disconnect();
|
|
}
|
|
|
|
bool ModemManager::ModemExists(const std::string& path) const {
|
|
CHECK(service_connected_);
|
|
if (ContainsKey(modems_, path)) {
|
|
LOG(INFO) << "ModemExists: " << path << " already exists.";
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void ModemManager::RecordAddedModem(shared_ptr<Modem> modem) {
|
|
modems_[modem->path()] = modem;
|
|
}
|
|
|
|
void ModemManager::RemoveModem(const string& path) {
|
|
LOG(INFO) << "Remove modem: " << path;
|
|
CHECK(service_connected_);
|
|
modems_.erase(path);
|
|
}
|
|
|
|
void ModemManager::OnDeviceInfoAvailable(const string& link_name) {
|
|
for (Modems::const_iterator it = modems_.begin(); it != modems_.end(); ++it) {
|
|
it->second->OnDeviceInfoAvailable(link_name);
|
|
}
|
|
}
|
|
|
|
// ModemManagerClassic
|
|
ModemManagerClassic::ModemManagerClassic(
|
|
ControlInterface* control_interface,
|
|
const string& service,
|
|
const string& path,
|
|
ModemInfo* modem_info)
|
|
: ModemManager(control_interface, service, path, modem_info) {}
|
|
|
|
ModemManagerClassic::~ModemManagerClassic() {
|
|
Stop();
|
|
}
|
|
|
|
void ModemManagerClassic::Start() {
|
|
LOG(INFO) << "Start watching modem manager service: " << service();
|
|
CHECK(!proxy_);
|
|
proxy_.reset(
|
|
control_interface()->CreateModemManagerProxy(
|
|
this,
|
|
path(),
|
|
service(),
|
|
base::Bind(&ModemManagerClassic::OnAppeared, base::Unretained(this)),
|
|
base::Bind(&ModemManagerClassic::OnVanished,
|
|
base::Unretained(this))));
|
|
}
|
|
|
|
void ModemManagerClassic::Stop() {
|
|
LOG(INFO) << "Stop watching modem manager service: " << service();
|
|
proxy_.reset();
|
|
Disconnect();
|
|
}
|
|
|
|
void ModemManagerClassic::Connect() {
|
|
ModemManager::Connect();
|
|
// TODO(petkov): Switch to asynchronous calls (crbug.com/200687).
|
|
vector<string> devices = proxy_->EnumerateDevices();
|
|
|
|
for (vector<string>::const_iterator it = devices.begin();
|
|
it != devices.end(); ++it) {
|
|
AddModemClassic(*it);
|
|
}
|
|
}
|
|
|
|
void ModemManagerClassic::AddModemClassic(const string& path) {
|
|
if (ModemExists(path)) {
|
|
return;
|
|
}
|
|
shared_ptr<ModemClassic> modem(new ModemClassic(service(),
|
|
path,
|
|
modem_info(),
|
|
control_interface()));
|
|
RecordAddedModem(modem);
|
|
InitModemClassic(modem);
|
|
}
|
|
|
|
void ModemManagerClassic::Disconnect() {
|
|
ModemManager::Disconnect();
|
|
}
|
|
|
|
void ModemManagerClassic::InitModemClassic(shared_ptr<ModemClassic> modem) {
|
|
// TODO(rochberg): Switch to asynchronous calls (crbug.com/200687).
|
|
if (modem == nullptr) {
|
|
return;
|
|
}
|
|
|
|
std::unique_ptr<DBusPropertiesProxyInterface> properties_proxy(
|
|
control_interface()->CreateDBusPropertiesProxy(modem->path(),
|
|
modem->service()));
|
|
KeyValueStore properties =
|
|
properties_proxy->GetAll(MM_MODEM_INTERFACE);
|
|
|
|
modem->CreateDeviceClassic(properties);
|
|
}
|
|
|
|
void ModemManagerClassic::OnDeviceAdded(const string& path) {
|
|
AddModemClassic(path);
|
|
}
|
|
|
|
void ModemManagerClassic::OnDeviceRemoved(const string& path) {
|
|
RemoveModem(path);
|
|
}
|
|
|
|
} // namespace shill
|