Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(775)

Unified Diff: net/base/network_change_notifier_linux.cc

Issue 11359141: Use Netlink instead of NetworkManager (via D-bus) to monitor network (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Address szym's second round of unittest comments Created 8 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « net/base/network_change_notifier_linux.h ('k') | net/base/network_change_notifier_linux_unittest.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: net/base/network_change_notifier_linux.cc
diff --git a/net/base/network_change_notifier_linux.cc b/net/base/network_change_notifier_linux.cc
index a3753a96c1e27df910e751ae88a95d21ddbedf3f..fc2b3720f9d61d6a842dc17f991ba646165c8a0b 100644
--- a/net/base/network_change_notifier_linux.cc
+++ b/net/base/network_change_notifier_linux.cc
@@ -1,252 +1,26 @@
// 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 <resolv.h>
-
#include "base/bind.h"
-#include "base/bind_helpers.h"
-#include "base/callback.h"
#include "base/compiler_specific.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/address_tracker_linux.h"
-#include "net/base/net_errors.h"
#include "net/dns/dns_config_service.h"
namespace net {
-namespace {
-
-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:
- explicit Thread(dbus::Bus* bus);
+ Thread();
virtual ~Thread();
// Plumbing for NetworkChangeNotifier::GetCurrentConnectionType.
// Safe to call from any thread.
NetworkChangeNotifier::ConnectionType GetCurrentConnectionType() {
- return network_manager_api_.GetCurrentConnectionType();
+ return address_tracker_.GetCurrentConnectionType();
}
const internal::AddressTrackerLinux* address_tracker() const {
@@ -259,24 +33,20 @@ class NetworkChangeNotifierLinux::Thread : public base::Thread {
virtual void CleanUp() OVERRIDE;
private:
- // Used to detect online/offline state changes.
- NetworkManagerApi network_manager_api_;
-
scoped_ptr<DnsConfigService> dns_config_service_;
+ // Used to detect online/offline state and IP address changes.
internal::AddressTrackerLinux address_tracker_;
DISALLOW_COPY_AND_ASSIGN(Thread);
};
-NetworkChangeNotifierLinux::Thread::Thread(dbus::Bus* bus)
+NetworkChangeNotifierLinux::Thread::Thread()
: base::Thread("NetworkChangeNotifier"),
- network_manager_api_(
- base::Bind(&NetworkChangeNotifier::
- NotifyObserversOfConnectionTypeChange),
- bus),
address_tracker_(
base::Bind(&NetworkChangeNotifier::
- NotifyObserversOfIPAddressChange)) {
+ NotifyObserversOfIPAddressChange),
+ base::Bind(&NetworkChangeNotifier::
+ NotifyObserversOfConnectionTypeChange)) {
}
NetworkChangeNotifierLinux::Thread::~Thread() {
@@ -284,7 +54,6 @@ NetworkChangeNotifierLinux::Thread::~Thread() {
}
void NetworkChangeNotifierLinux::Thread::Init() {
- network_manager_api_.Init();
address_tracker_.Init();
dns_config_service_ = DnsConfigService::CreateSystemService();
dns_config_service_->WatchConfig(
@@ -292,21 +61,15 @@ void NetworkChangeNotifierLinux::Thread::Init() {
}
void NetworkChangeNotifierLinux::Thread::CleanUp() {
- network_manager_api_.CleanUp();
dns_config_service_.reset();
}
NetworkChangeNotifierLinux* NetworkChangeNotifierLinux::Create() {
- return new NetworkChangeNotifierLinux(NULL);
-}
-
-NetworkChangeNotifierLinux* NetworkChangeNotifierLinux::CreateForTest(
- dbus::Bus* bus) {
- return new NetworkChangeNotifierLinux(bus);
+ return new NetworkChangeNotifierLinux();
}
-NetworkChangeNotifierLinux::NetworkChangeNotifierLinux(dbus::Bus* bus)
- : notifier_thread_(new Thread(bus)) {
+NetworkChangeNotifierLinux::NetworkChangeNotifierLinux()
+ : notifier_thread_(new Thread()) {
// We create this notifier thread because the notification implementation
// needs a MessageLoopForIO, and there's no guarantee that
// MessageLoop::current() meets that criterion.
« no previous file with comments | « net/base/network_change_notifier_linux.h ('k') | net/base/network_change_notifier_linux_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698