blob: 009dea11a3950e1659e2bed766732f32601a22c1 [file] [log] [blame]
Darin Petkov41c0e0a2012-01-09 16:38:53 +01001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Darin Petkov887f2982011-07-14 16:10:17 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "shill/modem_manager.h"
6
7#include <base/logging.h>
Darin Petkov5c97ac52011-07-19 16:30:49 -07008#include <base/stl_util-inl.h>
Darin Petkov887f2982011-07-14 16:10:17 -07009
Darin Petkov5c97ac52011-07-19 16:30:49 -070010#include "shill/modem.h"
Darin Petkovc90fe522011-07-15 13:59:47 -070011#include "shill/modem_manager_proxy.h"
12#include "shill/proxy_factory.h"
13
Darin Petkov887f2982011-07-14 16:10:17 -070014using std::string;
Darin Petkov5c97ac52011-07-19 16:30:49 -070015using std::tr1::shared_ptr;
16using std::vector;
Darin Petkov887f2982011-07-14 16:10:17 -070017
18namespace shill {
19
20ModemManager::ModemManager(const string &service,
21 const string &path,
22 ControlInterface *control_interface,
23 EventDispatcher *dispatcher,
Thieu Le3426c8f2012-01-11 17:35:11 -080024 Metrics *metrics,
Darin Petkov887f2982011-07-14 16:10:17 -070025 Manager *manager,
Darin Petkov137884a2011-10-26 18:52:47 +020026 GLib *glib,
27 mobile_provider_db *provider_db)
Darin Petkovab565bb2011-10-06 02:55:51 -070028 : proxy_factory_(ProxyFactory::GetInstance()),
29 service_(service),
Darin Petkov887f2982011-07-14 16:10:17 -070030 path_(path),
31 watcher_id_(0),
32 control_interface_(control_interface),
33 dispatcher_(dispatcher),
Thieu Le3426c8f2012-01-11 17:35:11 -080034 metrics_(metrics),
Darin Petkov67d8ecf2011-07-26 16:03:30 -070035 manager_(manager),
Darin Petkov137884a2011-10-26 18:52:47 +020036 glib_(glib),
37 provider_db_(provider_db) {}
Darin Petkov887f2982011-07-14 16:10:17 -070038
39ModemManager::~ModemManager() {
40 Stop();
41}
42
43void ModemManager::Start() {
44 LOG(INFO) << "Start watching modem manager service: " << service_;
Eric Shienbroodc74cf9c2012-03-02 15:00:35 -050045 CHECK_EQ(0U, watcher_id_);
Darin Petkovc90fe522011-07-15 13:59:47 -070046 // TODO(petkov): Implement DBus name watching through dbus-c++.
Darin Petkov887f2982011-07-14 16:10:17 -070047 watcher_id_ = glib_->BusWatchName(G_BUS_TYPE_SYSTEM,
48 service_.c_str(),
49 G_BUS_NAME_WATCHER_FLAGS_NONE,
50 OnAppear,
51 OnVanish,
52 this,
53 NULL);
54}
55
56void ModemManager::Stop() {
57 LOG(INFO) << "Stop watching modem manager service: " << service_;
58 if (watcher_id_) {
59 glib_->BusUnwatchName(watcher_id_);
60 watcher_id_ = 0;
61 }
62 Disconnect();
63}
64
65void ModemManager::Connect(const string &owner) {
David Rochberg81030ea2012-03-02 15:44:25 -050066 // Inheriting classes call this superclass method.
Darin Petkov887f2982011-07-14 16:10:17 -070067 owner_ = owner;
Darin Petkov887f2982011-07-14 16:10:17 -070068}
69
70void ModemManager::Disconnect() {
David Rochberg81030ea2012-03-02 15:44:25 -050071 // Inheriting classes call this superclass method.
Darin Petkov5c97ac52011-07-19 16:30:49 -070072 modems_.clear();
Darin Petkov887f2982011-07-14 16:10:17 -070073 owner_.clear();
Darin Petkov887f2982011-07-14 16:10:17 -070074}
75
mukesh agrawal1830fa12011-09-26 14:31:40 -070076void ModemManager::OnAppear(GDBusConnection */*connection*/,
Darin Petkov887f2982011-07-14 16:10:17 -070077 const gchar *name,
78 const gchar *name_owner,
79 gpointer user_data) {
80 LOG(INFO) << "Modem manager " << name << " appeared. Owner: " << name_owner;
81 ModemManager *manager = reinterpret_cast<ModemManager *>(user_data);
82 manager->Connect(name_owner);
83}
84
mukesh agrawal1830fa12011-09-26 14:31:40 -070085void ModemManager::OnVanish(GDBusConnection */*connection*/,
Darin Petkov887f2982011-07-14 16:10:17 -070086 const gchar *name,
87 gpointer user_data) {
88 LOG(INFO) << "Modem manager " << name << " vanished.";
89 ModemManager *manager = reinterpret_cast<ModemManager *>(user_data);
90 manager->Disconnect();
91}
92
Darin Petkov41c0e0a2012-01-09 16:38:53 +010093void ModemManager::AddModem(const string &path) {
Darin Petkov5c97ac52011-07-19 16:30:49 -070094 LOG(INFO) << "Add modem: " << path;
95 CHECK(!owner_.empty());
96 if (ContainsKey(modems_, path)) {
97 LOG(INFO) << "Modem already exists; ignored.";
98 return;
99 }
Darin Petkov137884a2011-10-26 18:52:47 +0200100 shared_ptr<Modem> modem(new Modem(owner_,
101 path,
102 control_interface_,
103 dispatcher_,
Thieu Le3426c8f2012-01-11 17:35:11 -0800104 metrics_,
Darin Petkov137884a2011-10-26 18:52:47 +0200105 manager_,
106 provider_db_));
Darin Petkov5c97ac52011-07-19 16:30:49 -0700107 modems_[path] = modem;
David Rochberg81030ea2012-03-02 15:44:25 -0500108 InitModem(modem);
Darin Petkov5c97ac52011-07-19 16:30:49 -0700109}
110
Darin Petkov41c0e0a2012-01-09 16:38:53 +0100111void ModemManager::RemoveModem(const string &path) {
Darin Petkov5c97ac52011-07-19 16:30:49 -0700112 LOG(INFO) << "Remove modem: " << path;
113 CHECK(!owner_.empty());
114 modems_.erase(path);
115}
116
Darin Petkov41c0e0a2012-01-09 16:38:53 +0100117void ModemManager::OnDeviceInfoAvailable(const string &link_name) {
David Rochberg81030ea2012-03-02 15:44:25 -0500118 for (Modems::const_iterator it = modems_.begin(); it != modems_.end(); ++it) {
Darin Petkov41c0e0a2012-01-09 16:38:53 +0100119 it->second->OnDeviceInfoAvailable(link_name);
120 }
121}
122
David Rochberg81030ea2012-03-02 15:44:25 -0500123// ModemManagerClassic
124ModemManagerClassic::ModemManagerClassic(const std::string &service,
125 const std::string &path,
126 ControlInterface *control_interface,
127 EventDispatcher *dispatcher,
128 Metrics *metrics,
129 Manager *manager,
130 GLib *glib,
131 mobile_provider_db *provider_db) :
132 ModemManager(service,
133 path,
134 control_interface,
135 dispatcher,
136 metrics,
137 manager,
138 glib,
139 provider_db) {}
140
141ModemManagerClassic::~ModemManagerClassic() {}
142
143void ModemManagerClassic::Connect(const string &supplied_owner) {
144 ModemManager::Connect(supplied_owner);
145 proxy_.reset(proxy_factory()->CreateModemManagerProxy(this, path(), owner()));
146
147 // TODO(petkov): Switch to asynchronous calls (crosbug.com/17583).
148 vector<DBus::Path> devices = proxy_->EnumerateDevices();
149 for (vector<DBus::Path>::const_iterator it = devices.begin();
150 it != devices.end(); ++it) {
151 AddModem(*it);
152 }
153}
154
155void ModemManagerClassic::Disconnect() {
156 ModemManager::Disconnect();
157 proxy_.reset();
158}
159
160void ModemManagerClassic::InitModem(shared_ptr<Modem> modem) {
161 modem->Init();
162}
163
Darin Petkov887f2982011-07-14 16:10:17 -0700164} // namespace shill