Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "net/base/network_change_notifier_mac.h" | 5 #include "net/base/network_change_notifier_mac.h" |
| 6 | 6 |
| 7 #include <netinet/in.h> | 7 #include <netinet/in.h> |
| 8 #include <resolv.h> | 8 #include <resolv.h> |
| 9 | 9 |
| 10 #include "base/macros.h" | 10 #include "base/macros.h" |
| 11 #include "base/message_loop/message_loop.h" | 11 #include "base/message_loop/message_loop.h" |
| 12 #include "base/task_runner_util.h" | |
| 12 #include "base/threading/thread.h" | 13 #include "base/threading/thread.h" |
| 13 #include "base/threading/thread_restrictions.h" | 14 #include "base/threading/thread_restrictions.h" |
| 14 #include "net/dns/dns_config_service.h" | 15 #include "net/dns/dns_config_service.h" |
| 15 | 16 |
| 16 namespace net { | 17 namespace net { |
| 17 | 18 |
| 18 static bool CalculateReachability(SCNetworkConnectionFlags flags) { | 19 static bool CalculateReachability(SCNetworkConnectionFlags flags) { |
| 19 bool reachable = flags & kSCNetworkFlagsReachable; | 20 bool reachable = flags & kSCNetworkFlagsReachable; |
| 20 bool connection_required = flags & kSCNetworkFlagsConnectionRequired; | 21 bool connection_required = flags & kSCNetworkFlagsConnectionRequired; |
| 21 return reachable && !connection_required; | 22 return reachable && !connection_required; |
| 22 } | 23 } |
| 23 | 24 |
| 24 NetworkChangeNotifier::ConnectionType CalculateConnectionType( | |
| 25 SCNetworkConnectionFlags flags) { | |
| 26 bool reachable = CalculateReachability(flags); | |
| 27 if (reachable) { | |
| 28 #if defined(OS_IOS) | |
| 29 return (flags & kSCNetworkReachabilityFlagsIsWWAN) ? | |
| 30 NetworkChangeNotifier::CONNECTION_3G : | |
| 31 NetworkChangeNotifier::CONNECTION_WIFI; | |
| 32 #else | |
| 33 // TODO(droger): Get something more detailed than CONNECTION_UNKNOWN. | |
| 34 // http://crbug.com/112937 | |
| 35 return NetworkChangeNotifier::CONNECTION_UNKNOWN; | |
| 36 #endif // defined(OS_IOS) | |
| 37 } else { | |
| 38 return NetworkChangeNotifier::CONNECTION_NONE; | |
| 39 } | |
| 40 } | |
| 41 | |
| 42 // Thread on which we can run DnsConfigService, which requires a TYPE_IO | 25 // Thread on which we can run DnsConfigService, which requires a TYPE_IO |
| 43 // message loop. | 26 // message loop. |
| 44 class NetworkChangeNotifierMac::DnsConfigServiceThread : public base::Thread { | 27 class NetworkChangeNotifierMac::DnsConfigServiceThread : public base::Thread { |
| 45 public: | 28 public: |
| 46 DnsConfigServiceThread() : base::Thread("DnsConfigService") {} | 29 DnsConfigServiceThread() : base::Thread("DnsConfigService") {} |
| 47 | 30 |
| 48 ~DnsConfigServiceThread() override { Stop(); } | 31 ~DnsConfigServiceThread() override { Stop(); } |
| 49 | 32 |
| 50 void Init() override { | 33 void Init() override { |
| 51 service_ = DnsConfigService::CreateSystemService(); | 34 service_ = DnsConfigService::CreateSystemService(); |
| 52 service_->WatchConfig(base::Bind(&NetworkChangeNotifier::SetDnsConfig)); | 35 service_->WatchConfig(base::Bind(&NetworkChangeNotifier::SetDnsConfig)); |
| 53 } | 36 } |
| 54 | 37 |
| 55 void CleanUp() override { service_.reset(); } | 38 void CleanUp() override { service_.reset(); } |
| 56 | 39 |
| 57 private: | 40 private: |
| 58 std::unique_ptr<DnsConfigService> service_; | 41 std::unique_ptr<DnsConfigService> service_; |
| 59 | 42 |
| 60 DISALLOW_COPY_AND_ASSIGN(DnsConfigServiceThread); | 43 DISALLOW_COPY_AND_ASSIGN(DnsConfigServiceThread); |
| 61 }; | 44 }; |
| 62 | 45 |
| 63 NetworkChangeNotifierMac::NetworkChangeNotifierMac() | 46 NetworkChangeNotifierMac::NetworkChangeNotifierMac() |
| 64 : NetworkChangeNotifier(NetworkChangeCalculatorParamsMac()), | 47 : NetworkChangeNotifier(NetworkChangeCalculatorParamsMac()), |
| 65 connection_type_(CONNECTION_UNKNOWN), | 48 connection_type_(CONNECTION_UNKNOWN), |
| 66 connection_type_initialized_(false), | 49 connection_type_initialized_(false), |
| 67 initial_connection_type_cv_(&connection_type_lock_), | 50 initial_connection_type_cv_(&connection_type_lock_), |
| 68 forwarder_(this), | 51 forwarder_(this), |
| 69 dns_config_service_thread_(new DnsConfigServiceThread()) { | 52 dns_config_service_thread_(new DnsConfigServiceThread()), |
| 53 weak_ptr_factory_(this) { | |
| 70 // Must be initialized after the rest of this object, as it may call back into | 54 // Must be initialized after the rest of this object, as it may call back into |
| 71 // SetInitialConnectionType(). | 55 // SetInitialConnectionType(). |
| 72 config_watcher_.reset(new NetworkConfigWatcherMac(&forwarder_)); | 56 config_watcher_.reset(new NetworkConfigWatcherMac(&forwarder_)); |
| 73 dns_config_service_thread_->StartWithOptions( | 57 dns_config_service_thread_->StartWithOptions( |
| 74 base::Thread::Options(base::MessageLoop::TYPE_IO, 0)); | 58 base::Thread::Options(base::MessageLoop::TYPE_IO, 0)); |
| 75 } | 59 } |
| 76 | 60 |
| 77 NetworkChangeNotifierMac::~NetworkChangeNotifierMac() { | 61 NetworkChangeNotifierMac::~NetworkChangeNotifierMac() { |
| 78 // Delete the ConfigWatcher to join the notifier thread, ensuring that | 62 // Delete the ConfigWatcher to join the notifier thread, ensuring that |
| 79 // StartReachabilityNotifications() has an opportunity to run to completion. | 63 // StartReachabilityNotifications() has an opportunity to run to completion. |
| (...skipping 15 matching lines...) Expand all Loading... | |
| 95 // Delay values arrived at by simple experimentation and adjusted so as to | 79 // Delay values arrived at by simple experimentation and adjusted so as to |
| 96 // produce a single signal when switching between network connections. | 80 // produce a single signal when switching between network connections. |
| 97 params.ip_address_offline_delay_ = base::TimeDelta::FromMilliseconds(500); | 81 params.ip_address_offline_delay_ = base::TimeDelta::FromMilliseconds(500); |
| 98 params.ip_address_online_delay_ = base::TimeDelta::FromMilliseconds(500); | 82 params.ip_address_online_delay_ = base::TimeDelta::FromMilliseconds(500); |
| 99 params.connection_type_offline_delay_ = | 83 params.connection_type_offline_delay_ = |
| 100 base::TimeDelta::FromMilliseconds(1000); | 84 base::TimeDelta::FromMilliseconds(1000); |
| 101 params.connection_type_online_delay_ = base::TimeDelta::FromMilliseconds(500); | 85 params.connection_type_online_delay_ = base::TimeDelta::FromMilliseconds(500); |
| 102 return params; | 86 return params; |
| 103 } | 87 } |
| 104 | 88 |
| 89 // static | |
| 90 NetworkChangeNotifier::ConnectionType | |
| 91 NetworkChangeNotifierMac::CalculateConnectionType( | |
| 92 SCNetworkConnectionFlags flags) { | |
| 93 // Called on notifier thread during forwarder initialization, otherwise on | |
| 94 // the dns thread. | |
| 95 bool reachable = CalculateReachability(flags); | |
| 96 if (reachable) { | |
| 97 #if defined(OS_IOS) | |
| 98 return (flags & kSCNetworkReachabilityFlagsIsWWAN) | |
| 99 ? NetworkChangeNotifier::CONNECTION_3G | |
| 100 : NetworkChangeNotifier::CONNECTION_WIFI; | |
| 101 #else | |
| 102 // TODO(droger): Get something more detailed than CONNECTION_UNKNOWN. | |
| 103 // http://crbug.com/112937 | |
|
pauljensen
2017/05/23 15:10:30
I think we can remove this comment now. Can you a
jkarlin
2017/05/27 01:06:19
Done.
| |
| 104 return ConnectionTypeFromInterfaces(); | |
| 105 #endif // defined(OS_IOS) | |
| 106 } else { | |
| 107 return CONNECTION_NONE; | |
| 108 } | |
| 109 } | |
| 110 | |
| 105 NetworkChangeNotifier::ConnectionType | 111 NetworkChangeNotifier::ConnectionType |
| 106 NetworkChangeNotifierMac::GetCurrentConnectionType() const { | 112 NetworkChangeNotifierMac::GetCurrentConnectionType() const { |
| 107 base::ThreadRestrictions::ScopedAllowWait allow_wait; | 113 base::ThreadRestrictions::ScopedAllowWait allow_wait; |
| 108 base::AutoLock lock(connection_type_lock_); | 114 base::AutoLock lock(connection_type_lock_); |
| 109 // Make sure the initial connection type is set before returning. | 115 // Make sure the initial connection type is set before returning. |
| 110 while (!connection_type_initialized_) { | 116 while (!connection_type_initialized_) { |
| 111 initial_connection_type_cv_.Wait(); | 117 initial_connection_type_cv_.Wait(); |
| 112 } | 118 } |
| 113 return connection_type_; | 119 return connection_type_; |
| 114 } | 120 } |
| (...skipping 126 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 241 #endif // defined(OS_IOS) | 247 #endif // defined(OS_IOS) |
| 242 } | 248 } |
| 243 | 249 |
| 244 // static | 250 // static |
| 245 void NetworkChangeNotifierMac::ReachabilityCallback( | 251 void NetworkChangeNotifierMac::ReachabilityCallback( |
| 246 SCNetworkReachabilityRef target, | 252 SCNetworkReachabilityRef target, |
| 247 SCNetworkConnectionFlags flags, | 253 SCNetworkConnectionFlags flags, |
| 248 void* notifier) { | 254 void* notifier) { |
| 249 NetworkChangeNotifierMac* notifier_mac = | 255 NetworkChangeNotifierMac* notifier_mac = |
| 250 static_cast<NetworkChangeNotifierMac*>(notifier); | 256 static_cast<NetworkChangeNotifierMac*>(notifier); |
| 251 | |
|
pauljensen
2017/05/23 15:10:30
whitespace change?
jkarlin
2017/05/27 01:06:19
Done.
| |
| 252 DCHECK_EQ(notifier_mac->run_loop_.get(), CFRunLoopGetCurrent()); | 257 DCHECK_EQ(notifier_mac->run_loop_.get(), CFRunLoopGetCurrent()); |
| 253 | 258 |
| 254 ConnectionType new_type = CalculateConnectionType(flags); | 259 base::PostTaskAndReplyWithResult( |
| 260 notifier_mac->dns_config_service_thread_->message_loop() | |
| 261 ->task_runner() | |
|
pauljensen
2017/05/23 15:18:15
hmm so I think this may not be the thread we want
jkarlin
2017/05/27 01:06:19
After F2F, moved to notifier thread and enabled bl
| |
| 262 .get(), | |
| 263 FROM_HERE, | |
| 264 base::Bind(&NetworkChangeNotifierMac::CalculateConnectionType, flags), | |
| 265 base::Bind(&NetworkChangeNotifierMac::ReachabilityCallbackImpl, | |
| 266 notifier_mac->weak_ptr_factory_.GetWeakPtr())); | |
| 267 } | |
| 268 | |
| 269 void NetworkChangeNotifierMac::ReachabilityCallbackImpl( | |
| 270 ConnectionType new_type) { | |
| 255 ConnectionType old_type; | 271 ConnectionType old_type; |
| 256 { | 272 { |
| 257 base::AutoLock lock(notifier_mac->connection_type_lock_); | 273 base::AutoLock lock(connection_type_lock_); |
| 258 old_type = notifier_mac->connection_type_; | 274 old_type = connection_type_; |
| 259 notifier_mac->connection_type_ = new_type; | 275 connection_type_ = new_type; |
| 260 } | 276 } |
| 261 if (old_type != new_type) { | 277 if (old_type != new_type) { |
| 262 NotifyObserversOfConnectionTypeChange(); | 278 NotifyObserversOfConnectionTypeChange(); |
| 263 double max_bandwidth_mbps = | 279 double max_bandwidth_mbps = GetMaxBandwidthForConnectionSubtype( |
| 264 NetworkChangeNotifier::GetMaxBandwidthForConnectionSubtype( | 280 new_type == CONNECTION_NONE ? SUBTYPE_NONE : SUBTYPE_UNKNOWN); |
| 265 new_type == CONNECTION_NONE ? SUBTYPE_NONE : SUBTYPE_UNKNOWN); | |
| 266 NotifyObserversOfMaxBandwidthChange(max_bandwidth_mbps, new_type); | 281 NotifyObserversOfMaxBandwidthChange(max_bandwidth_mbps, new_type); |
| 267 } | 282 } |
| 268 | 283 |
| 269 #if defined(OS_IOS) | 284 #if defined(OS_IOS) |
| 270 // On iOS, the SCDynamicStore API does not exist, and we use the reachability | 285 // On iOS, the SCDynamicStore API does not exist, and we use the reachability |
| 271 // API to detect IP address changes instead. | 286 // API to detect IP address changes instead. |
| 272 NotifyObserversOfIPAddressChange(); | 287 NotifyObserversOfIPAddressChange(); |
| 273 #endif // defined(OS_IOS) | 288 #endif // defined(OS_IOS) |
| 274 } | 289 } |
| 275 | 290 |
| 276 } // namespace net | 291 } // namespace net |
| OLD | NEW |