blob: c5b1cb4b481b9c28f579dd67d27acc425fbbde6a [file] [log] [blame]
//
// Copyright (C) 2014 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 "apmanager/manager.h"
#include <base/bind.h>
#if !defined(__ANDROID__)
#include <chromeos/dbus/service_constants.h>
#else
#include "dbus/apmanager/dbus-constants.h"
#endif // __ANDROID__
using brillo::dbus_utils::AsyncEventSequencer;
using brillo::dbus_utils::ExportedObjectManager;
using brillo::dbus_utils::DBusMethodResponse;
using std::string;
namespace apmanager {
Manager::Manager(ControlInterface* control_interface)
: org::chromium::apmanager::ManagerAdaptor(this),
control_interface_(control_interface),
service_identifier_(0),
device_info_(this) {}
Manager::~Manager() {
// Terminate all services before cleanup other resources.
for (auto& service : services_) {
service.reset();
}
}
void Manager::RegisterAsync(
ExportedObjectManager* object_manager,
const scoped_refptr<dbus::Bus>& bus,
const base::Callback<void(bool)>& completion_callback) {
CHECK(!dbus_object_) << "Already registered";
dbus_object_.reset(
new brillo::dbus_utils::DBusObject(
object_manager,
bus,
org::chromium::apmanager::ManagerAdaptor::GetObjectPath()));
RegisterWithDBusObject(dbus_object_.get());
dbus_object_->RegisterAsync(completion_callback);
bus_ = bus;
shill_manager_.Init(control_interface_);
firewall_manager_.Init(control_interface_);
}
void Manager::Start() {
device_info_.Start();
}
void Manager::Stop() {
device_info_.Stop();
}
bool Manager::CreateService(brillo::ErrorPtr* error,
dbus::Message* message,
dbus::ObjectPath* out_service) {
LOG(INFO) << "Manager::CreateService";
std::unique_ptr<Service> service(new Service(this, service_identifier_));
*out_service = service->adaptor()->GetRpcObjectIdentifier();
services_.push_back(std::move(service));
base::Closure on_connection_vanish = base::Bind(
&Manager::OnAPServiceOwnerDisappeared,
base::Unretained(this),
service_identifier_);
service_watchers_[service_identifier_].reset(
new DBusServiceWatcher{bus_, message->GetSender(), on_connection_vanish});
service_identifier_++;
return true;
}
bool Manager::RemoveService(brillo::ErrorPtr* error,
dbus::Message* message,
const dbus::ObjectPath& in_service) {
for (auto it = services_.begin(); it != services_.end(); ++it) {
if ((*it)->adaptor()->GetRpcObjectIdentifier() == in_service) {
// Verify the owner.
auto watcher = service_watchers_.find((*it)->identifier());
CHECK(watcher != service_watchers_.end())
<< "DBus watcher not created for service: " << (*it)->identifier();
if (watcher->second->connection_name() != message->GetSender()) {
brillo::Error::AddToPrintf(
error, FROM_HERE, brillo::errors::dbus::kDomain, kManagerError,
"Service %d is owned by another local process.",
(*it)->identifier());
return false;
}
service_watchers_.erase(watcher);
services_.erase(it);
return true;
}
}
brillo::Error::AddTo(
error, FROM_HERE, brillo::errors::dbus::kDomain, kManagerError,
"Service does not exist");
return false;
}
scoped_refptr<Device> Manager::GetAvailableDevice() {
for (const auto& device : devices_) {
// Look for an unused device with AP interface mode support.
if (!device->GetInUse() && !device->GetPreferredApInterface().empty()) {
return device;
}
}
return nullptr;
}
scoped_refptr<Device> Manager::GetDeviceFromInterfaceName(
const string& interface_name) {
for (const auto& device : devices_) {
if (device->InterfaceExists(interface_name)) {
return device;
}
}
return nullptr;
}
void Manager::RegisterDevice(scoped_refptr<Device> device) {
LOG(INFO) << "Manager::RegisterDevice: registering device "
<< device->GetDeviceName();
devices_.push_back(device);
// TODO(zqiu): Property update for available devices.
}
void Manager::ClaimInterface(const string& interface_name) {
shill_manager_.ClaimInterface(interface_name);
}
void Manager::ReleaseInterface(const string& interface_name) {
shill_manager_.ReleaseInterface(interface_name);
}
#if defined(__BRILLO__)
bool Manager::SetupApModeInterface(string* interface_name) {
return shill_manager_.SetupApModeInterface(interface_name);
}
bool Manager::SetupStationModeInterface(string* interface_name) {
return shill_manager_.SetupStationModeInterface(interface_name);
}
#endif // __BRILLO__
void Manager::RequestDHCPPortAccess(const string& interface) {
firewall_manager_.RequestDHCPPortAccess(interface);
}
void Manager::ReleaseDHCPPortAccess(const string& interface) {
firewall_manager_.ReleaseDHCPPortAccess(interface);
}
void Manager::OnAPServiceOwnerDisappeared(int service_identifier) {
LOG(INFO) << "Owner for service " << service_identifier << " disappeared";
// Remove service watcher.
auto watcher = service_watchers_.find(service_identifier);
CHECK(watcher != service_watchers_.end())
<< "Owner disappeared without watcher setup";
service_watchers_.erase(watcher);
// Remove the service.
for (auto it = services_.begin(); it != services_.end(); ++it) {
if ((*it)->identifier() == service_identifier) {
services_.erase(it);
return;
}
}
LOG(INFO) << "Owner for service " << service_identifier
<< " disappeared before it is registered";
}
} // namespace apmanager