blob: b7acd468aedca8dd5757884e7a85ef336fc4078d [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// This implementation of NetworkChangeNotifier's offline state detection
// depends on D-Bus and NetworkManager, and is known to work on at least
// GNOME version 2.30. If D-Bus or NetworkManager are unavailable, this
// implementation will always behave as if it is online.
#include "net/base/network_change_notifier_linux.h"
#include <errno.h>
#include <resolv.h>
#include <sys/socket.h>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/callback.h"
#include "base/compiler_specific.h"
#include "base/eintr_wrapper.h"
#include "base/memory/weak_ptr.h"
#include "base/synchronization/lock.h"
#include "base/synchronization/waitable_event.h"
#include "base/threading/platform_thread.h"
#include "base/threading/thread.h"
#include "base/threading/thread_restrictions.h"
#include "dbus/bus.h"
#include "dbus/message.h"
#include "dbus/object_proxy.h"
#include "net/base/net_errors.h"
#include "net/base/network_change_notifier_netlink_linux.h"
#include "net/dns/dns_config_watcher.h"
namespace net {
namespace {
const int kInvalidSocket = -1;
const char kNetworkManagerServiceName[] = "org.freedesktop.NetworkManager";
const char kNetworkManagerPath[] = "/org/freedesktop/NetworkManager";
const char kNetworkManagerInterface[] = "org.freedesktop.NetworkManager";
// http://projects.gnome.org/NetworkManager/developers/spec-08.html#type-NM_STATE
enum {
NM_LEGACY_STATE_UNKNOWN = 0,
NM_LEGACY_STATE_ASLEEP = 1,
NM_LEGACY_STATE_CONNECTING = 2,
NM_LEGACY_STATE_CONNECTED = 3,
NM_LEGACY_STATE_DISCONNECTED = 4
};
// http://projects.gnome.org/NetworkManager/developers/migrating-to-09/spec.html#type-NM_STATE
enum {
NM_STATE_UNKNOWN = 0,
NM_STATE_ASLEEP = 10,
NM_STATE_DISCONNECTED = 20,
NM_STATE_DISCONNECTING = 30,
NM_STATE_CONNECTING = 40,
NM_STATE_CONNECTED_LOCAL = 50,
NM_STATE_CONNECTED_SITE = 60,
NM_STATE_CONNECTED_GLOBAL = 70
};
} // namespace
// A wrapper around NetworkManager's D-Bus API.
class NetworkManagerApi {
public:
NetworkManagerApi(const base::Closure& notification_callback, dbus::Bus* bus)
: is_offline_(false),
offline_state_initialized_(true /*manual_reset*/, false),
notification_callback_(notification_callback),
helper_thread_id_(base::kInvalidThreadId),
ALLOW_THIS_IN_INITIALIZER_LIST(ptr_factory_(this)),
system_bus_(bus) { }
~NetworkManagerApi() { }
// Should be called on a helper thread which must be of type IO.
void Init();
// Must be called by the helper thread's CleanUp() method.
void CleanUp();
// Implementation of NetworkChangeNotifierLinux::GetCurrentConnectionType().
// Safe to call from any thread, but will block until Init() has completed.
NetworkChangeNotifier::ConnectionType GetCurrentConnectionType();
private:
// Callbacks for D-Bus API.
void OnInitialResponse(dbus::Response* response) {
HandleResponse(response);
offline_state_initialized_.Signal();
}
void OnSignaled(dbus::Signal* signal);
void OnConnected(const std::string&, const std::string&, bool success) {
if (!success) {
DLOG(WARNING) << "Failed to set up offline state detection";
offline_state_initialized_.Signal();
}
}
// Helper for OnInitialResponse.
void HandleResponse(dbus::Response* response);
// Converts a NetworkManager state uint to a bool.
static bool StateIsOffline(uint32 state);
bool is_offline_;
base::Lock is_offline_lock_;
base::WaitableEvent offline_state_initialized_;
base::Closure notification_callback_;
base::PlatformThreadId helper_thread_id_;
base::WeakPtrFactory<NetworkManagerApi> ptr_factory_;
scoped_refptr<dbus::Bus> system_bus_;
DISALLOW_COPY_AND_ASSIGN(NetworkManagerApi);
};
void NetworkManagerApi::Init() {
// D-Bus requires an IO MessageLoop.
DCHECK_EQ(MessageLoop::current()->type(), MessageLoop::TYPE_IO);
helper_thread_id_ = base::PlatformThread::CurrentId();
if (!system_bus_) {
dbus::Bus::Options options;
options.bus_type = dbus::Bus::SYSTEM;
options.connection_type = dbus::Bus::PRIVATE;
system_bus_ = new dbus::Bus(options);
}
// Ignore ServiceUnknown errors to avoid log spam: http://crbug.com/109696.
dbus::ObjectProxy* proxy = system_bus_->GetObjectProxyWithOptions(
kNetworkManagerServiceName, dbus::ObjectPath(kNetworkManagerPath),
dbus::ObjectProxy::IGNORE_SERVICE_UNKNOWN_ERRORS);
// Get the initial state asynchronously.
dbus::MethodCall method_call(DBUS_INTERFACE_PROPERTIES, "Get");
dbus::MessageWriter builder(&method_call);
builder.AppendString(kNetworkManagerInterface);
builder.AppendString("State");
proxy->CallMethod(
&method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT,
base::Bind(&NetworkManagerApi::OnInitialResponse,
ptr_factory_.GetWeakPtr()));
// And sign up for notifications.
proxy->ConnectToSignal(
kNetworkManagerInterface,
"StateChanged",
base::Bind(&NetworkManagerApi::OnSignaled, ptr_factory_.GetWeakPtr()),
base::Bind(&NetworkManagerApi::OnConnected, ptr_factory_.GetWeakPtr()));
}
void NetworkManagerApi::CleanUp() {
DCHECK_EQ(helper_thread_id_, base::PlatformThread::CurrentId());
ptr_factory_.InvalidateWeakPtrs();
}
void NetworkManagerApi::HandleResponse(dbus::Response* response) {
DCHECK_EQ(helper_thread_id_, base::PlatformThread::CurrentId());
if (!response) {
DVLOG(1) << "No response received for initial state request";
return;
}
dbus::MessageReader reader(response);
uint32 state = 0;
if (!reader.PopVariantOfUint32(&state)) {
DLOG(WARNING) << "Unexpected response for NetworkManager State request: "
<< response->ToString();
return;
}
{
base::AutoLock lock(is_offline_lock_);
is_offline_ = StateIsOffline(state);
}
}
void NetworkManagerApi::OnSignaled(dbus::Signal* signal) {
DCHECK_EQ(helper_thread_id_, base::PlatformThread::CurrentId());
dbus::MessageReader reader(signal);
uint32 state = 0;
if (!reader.PopUint32(&state)) {
DLOG(WARNING) << "Unexpected signal for NetworkManager StateChanged: "
<< signal->ToString();
return;
}
bool new_is_offline = StateIsOffline(state);
{
base::AutoLock lock(is_offline_lock_);
if (is_offline_ != new_is_offline)
is_offline_ = new_is_offline;
else
return;
}
notification_callback_.Run();
}
bool NetworkManagerApi::StateIsOffline(uint32 state) {
switch (state) {
case NM_LEGACY_STATE_CONNECTED:
case NM_STATE_CONNECTED_SITE:
case NM_STATE_CONNECTED_GLOBAL:
// Definitely connected
return false;
case NM_LEGACY_STATE_DISCONNECTED:
case NM_STATE_DISCONNECTED:
// Definitely disconnected
return true;
case NM_STATE_CONNECTED_LOCAL:
// Local networking only; I'm treating this as offline (keybuk)
return true;
case NM_LEGACY_STATE_CONNECTING:
case NM_STATE_DISCONNECTING:
case NM_STATE_CONNECTING:
// In-flight change to connection status currently underway
return true;
case NM_LEGACY_STATE_ASLEEP:
case NM_STATE_ASLEEP:
// Networking disabled or no devices on system
return true;
default:
// Unknown status
return false;
}
}
NetworkChangeNotifier::ConnectionType
NetworkManagerApi::GetCurrentConnectionType() {
// http://crbug.com/125097
base::ThreadRestrictions::ScopedAllowWait allow_wait;
offline_state_initialized_.Wait();
base::AutoLock lock(is_offline_lock_);
// TODO(droger): Return something more detailed than CONNECTION_UNKNOWN.
return is_offline_ ? NetworkChangeNotifier::CONNECTION_NONE :
NetworkChangeNotifier::CONNECTION_UNKNOWN;
}
class NetworkChangeNotifierLinux::Thread
: public base::Thread, public MessageLoopForIO::Watcher {
public:
explicit Thread(dbus::Bus* bus);
virtual ~Thread();
// MessageLoopForIO::Watcher:
virtual void OnFileCanReadWithoutBlocking(int fd) OVERRIDE;
virtual void OnFileCanWriteWithoutBlocking(int /* fd */) OVERRIDE;
// Plumbing for NetworkChangeNotifier::GetCurrentConnectionType.
// Safe to call from any thread.
NetworkChangeNotifier::ConnectionType GetCurrentConnectionType() {
return network_manager_api_.GetCurrentConnectionType();
}
protected:
// base::Thread
virtual void Init() OVERRIDE;
virtual void CleanUp() OVERRIDE;
private:
// Starts listening for netlink messages. Also handles the messages if there
// are any available on the netlink socket.
void ListenForNotifications();
// Attempts to read from the netlink socket into |buf| of length |len|.
// Returns the bytes read on synchronous success and ERR_IO_PENDING if the
// recv() would block. Otherwise, it returns a net error code.
int ReadNotificationMessage(char* buf, size_t len);
// The netlink socket descriptor.
int netlink_fd_;
MessageLoopForIO::FileDescriptorWatcher netlink_watcher_;
// Used to detect online/offline state changes.
NetworkManagerApi network_manager_api_;
internal::DnsConfigWatcher dns_watcher_;
DISALLOW_COPY_AND_ASSIGN(Thread);
};
NetworkChangeNotifierLinux::Thread::Thread(dbus::Bus* bus)
: base::Thread("NetworkChangeNotifier"),
netlink_fd_(kInvalidSocket),
network_manager_api_(
base::Bind(&NetworkChangeNotifier::
NotifyObserversOfConnectionTypeChange),
bus) {
}
NetworkChangeNotifierLinux::Thread::~Thread() {
DCHECK(!Thread::IsRunning());
}
void NetworkChangeNotifierLinux::Thread::Init() {
netlink_fd_ = InitializeNetlinkSocket();
if (netlink_fd_ < 0) {
netlink_fd_ = kInvalidSocket;
return;
}
ListenForNotifications();
network_manager_api_.Init();
dns_watcher_.Init();
}
void NetworkChangeNotifierLinux::Thread::CleanUp() {
if (netlink_fd_ != kInvalidSocket) {
if (HANDLE_EINTR(close(netlink_fd_)) != 0)
PLOG(ERROR) << "Failed to close socket";
netlink_fd_ = kInvalidSocket;
netlink_watcher_.StopWatchingFileDescriptor();
}
network_manager_api_.CleanUp();
dns_watcher_.CleanUp();
}
void NetworkChangeNotifierLinux::Thread::OnFileCanReadWithoutBlocking(int fd) {
DCHECK_EQ(fd, netlink_fd_);
ListenForNotifications();
}
void NetworkChangeNotifierLinux::Thread::OnFileCanWriteWithoutBlocking(
int /* fd */) {
NOTREACHED();
}
void NetworkChangeNotifierLinux::Thread::ListenForNotifications() {
char buf[4096];
int rv = ReadNotificationMessage(buf, arraysize(buf));
while (rv > 0) {
if (HandleNetlinkMessage(buf, rv)) {
VLOG(1) << "Detected IP address changes.";
NotifyObserversOfIPAddressChange();
}
rv = ReadNotificationMessage(buf, arraysize(buf));
}
if (rv == ERR_IO_PENDING) {
rv = MessageLoopForIO::current()->WatchFileDescriptor(netlink_fd_, false,
MessageLoopForIO::WATCH_READ, &netlink_watcher_, this);
LOG_IF(ERROR, !rv) << "Failed to watch netlink socket: " << netlink_fd_;
}
}
int NetworkChangeNotifierLinux::Thread::ReadNotificationMessage(
char* buf,
size_t len) {
DCHECK_NE(len, 0u);
DCHECK(buf);
memset(buf, 0, len);
int rv = recv(netlink_fd_, buf, len, 0);
if (rv > 0)
return rv;
DCHECK_NE(rv, 0);
if (errno != EAGAIN && errno != EWOULDBLOCK) {
PLOG(DFATAL) << "recv";
return ERR_FAILED;
}
return ERR_IO_PENDING;
}
NetworkChangeNotifierLinux* NetworkChangeNotifierLinux::Create() {
return new NetworkChangeNotifierLinux(NULL);
}
NetworkChangeNotifierLinux* NetworkChangeNotifierLinux::CreateForTest(
dbus::Bus* bus) {
return new NetworkChangeNotifierLinux(bus);
}
NetworkChangeNotifierLinux::NetworkChangeNotifierLinux(dbus::Bus* bus)
: notifier_thread_(new Thread(bus)) {
// We create this notifier thread because the notification implementation
// needs a MessageLoopForIO, and there's no guarantee that
// MessageLoop::current() meets that criterion.
base::Thread::Options thread_options(MessageLoop::TYPE_IO, 0);
notifier_thread_->StartWithOptions(thread_options);
}
NetworkChangeNotifierLinux::~NetworkChangeNotifierLinux() {
// Stopping from here allows us to sanity- check that the notifier
// thread shut down properly.
notifier_thread_->Stop();
}
NetworkChangeNotifier::ConnectionType
NetworkChangeNotifierLinux::GetCurrentConnectionType() const {
return notifier_thread_->GetCurrentConnectionType();
}
} // namespace net