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

Side by Side Diff: net/base/network_change_notifier_mac.cc

Issue 7006015: Implement NetworkChangeNotifier::IsCurrentlyOffline() for Mac. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Copyrights Created 9 years, 6 months 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 unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « net/base/network_change_notifier_mac.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2011 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 <SystemConfiguration/SCDynamicStoreKey.h> 8 #include <SystemConfiguration/SCDynamicStoreKey.h>
9 #include <SystemConfiguration/SCNetworkReachability.h>
8 #include <SystemConfiguration/SCSchemaDefinitions.h> 10 #include <SystemConfiguration/SCSchemaDefinitions.h>
9 11
10 #include "base/mac/scoped_cftyperef.h" 12 #include "base/mac/scoped_cftyperef.h"
11 13
12 namespace net { 14 namespace net {
13 15
14 NetworkChangeNotifierMac::NetworkChangeNotifierMac() 16 NetworkChangeNotifierMac::NetworkChangeNotifierMac()
15 : forwarder_(this), 17 : forwarder_(this),
16 config_watcher_(&forwarder_) {} 18 config_watcher_(&forwarder_),
17 NetworkChangeNotifierMac::~NetworkChangeNotifierMac() {} 19 network_reachable_(true) {}
20
21 NetworkChangeNotifierMac::~NetworkChangeNotifierMac() {
22 if (reachability_.get() && run_loop_.get()) {
23 SCNetworkReachabilityUnscheduleFromRunLoop(reachability_.get(),
24 run_loop_.get(),
25 kCFRunLoopCommonModes);
26 }
27 }
18 28
19 bool NetworkChangeNotifierMac::IsCurrentlyOffline() const { 29 bool NetworkChangeNotifierMac::IsCurrentlyOffline() const {
20 // TODO(eroman): http://crbug.com/53473 30 return !network_reachable_;
21 return false;
22 } 31 }
23 32
24 void NetworkChangeNotifierMac::SetDynamicStoreNotificationKeys( 33 void NetworkChangeNotifierMac::SetDynamicStoreNotificationKeys(
25 SCDynamicStoreRef store) { 34 SCDynamicStoreRef store) {
26 // Called on notifier thread. 35 // Called on notifier thread.
36 run_loop_.reset(CFRunLoopGetCurrent());
37 CFRetain(run_loop_.get());
38
27 base::mac::ScopedCFTypeRef<CFMutableArrayRef> notification_keys( 39 base::mac::ScopedCFTypeRef<CFMutableArrayRef> notification_keys(
28 CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks)); 40 CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks));
29 base::mac::ScopedCFTypeRef<CFStringRef> key( 41 base::mac::ScopedCFTypeRef<CFStringRef> key(
30 SCDynamicStoreKeyCreateNetworkGlobalEntity( 42 SCDynamicStoreKeyCreateNetworkGlobalEntity(
31 NULL, kSCDynamicStoreDomainState, kSCEntNetInterface)); 43 NULL, kSCDynamicStoreDomainState, kSCEntNetInterface));
32 CFArrayAppendValue(notification_keys.get(), key.get()); 44 CFArrayAppendValue(notification_keys.get(), key.get());
33 key.reset(SCDynamicStoreKeyCreateNetworkGlobalEntity( 45 key.reset(SCDynamicStoreKeyCreateNetworkGlobalEntity(
34 NULL, kSCDynamicStoreDomainState, kSCEntNetIPv4)); 46 NULL, kSCDynamicStoreDomainState, kSCEntNetIPv4));
35 CFArrayAppendValue(notification_keys.get(), key.get()); 47 CFArrayAppendValue(notification_keys.get(), key.get());
36 key.reset(SCDynamicStoreKeyCreateNetworkGlobalEntity( 48 key.reset(SCDynamicStoreKeyCreateNetworkGlobalEntity(
37 NULL, kSCDynamicStoreDomainState, kSCEntNetIPv6)); 49 NULL, kSCDynamicStoreDomainState, kSCEntNetIPv6));
38 CFArrayAppendValue(notification_keys.get(), key.get()); 50 CFArrayAppendValue(notification_keys.get(), key.get());
39 51
40 // Set the notification keys. This starts us receiving notifications. 52 // Set the notification keys. This starts us receiving notifications.
41 bool ret = SCDynamicStoreSetNotificationKeys( 53 bool ret = SCDynamicStoreSetNotificationKeys(
42 store, notification_keys.get(), NULL); 54 store, notification_keys.get(), NULL);
43 // TODO(willchan): Figure out a proper way to handle this rather than crash. 55 // TODO(willchan): Figure out a proper way to handle this rather than crash.
44 CHECK(ret); 56 CHECK(ret);
57
58 // Try to reach 0.0.0.0. This is the approach taken by Firefox:
59 //
60 // http://mxr.mozilla.org/mozilla2.0/source/netwerk/system/mac/nsNetworkLinkSe rvice.mm
61 //
62 // From my (adamk) testing on Snow Leopard, 0.0.0.0
63 // seems to be reachable if any network connection is available.
64 struct sockaddr_in addr = {0};
65 addr.sin_len = sizeof(addr);
66 addr.sin_family = AF_INET;
67 reachability_.reset(SCNetworkReachabilityCreateWithAddress(
68 kCFAllocatorDefault, reinterpret_cast<struct sockaddr*>(&addr)));
69 SCNetworkReachabilityContext reachability_context = {
70 0, // version
71 this, // user data
72 NULL, // retain
73 NULL, // release
74 NULL // description
75 };
76 if (!SCNetworkReachabilitySetCallback(
77 reachability_.get(),
78 &NetworkChangeNotifierMac::ReachabilityCallback,
79 &reachability_context)) {
80 LOG(DFATAL) << "Could not set network reachability callback";
81 reachability_.reset();
82 } else if (!SCNetworkReachabilityScheduleWithRunLoop(reachability_.get(),
83 run_loop_,
84 kCFRunLoopCommonModes)) {
85 LOG(DFATAL) << "Could not schedule network reachability on run loop";
86 reachability_.reset();
87 }
45 } 88 }
46 89
47 void NetworkChangeNotifierMac::OnNetworkConfigChange(CFArrayRef changed_keys) { 90 void NetworkChangeNotifierMac::OnNetworkConfigChange(CFArrayRef changed_keys) {
48 // Called on notifier thread. 91 DCHECK_EQ(run_loop_.get(), CFRunLoopGetCurrent());
49 92
50 for (CFIndex i = 0; i < CFArrayGetCount(changed_keys); ++i) { 93 for (CFIndex i = 0; i < CFArrayGetCount(changed_keys); ++i) {
51 CFStringRef key = static_cast<CFStringRef>( 94 CFStringRef key = static_cast<CFStringRef>(
52 CFArrayGetValueAtIndex(changed_keys, i)); 95 CFArrayGetValueAtIndex(changed_keys, i));
53 if (CFStringHasSuffix(key, kSCEntNetIPv4) || 96 if (CFStringHasSuffix(key, kSCEntNetIPv4) ||
54 CFStringHasSuffix(key, kSCEntNetIPv6)) { 97 CFStringHasSuffix(key, kSCEntNetIPv6)) {
55 NotifyObserversOfIPAddressChange(); 98 NotifyObserversOfIPAddressChange();
56 return; 99 return;
57 } 100 }
58 if (CFStringHasSuffix(key, kSCEntNetInterface)) { 101 if (CFStringHasSuffix(key, kSCEntNetInterface)) {
59 // TODO(willchan): Does not appear to be working. Look into this. 102 // TODO(willchan): Does not appear to be working. Look into this.
60 // Perhaps this isn't needed anyway. 103 // Perhaps this isn't needed anyway.
61 } else { 104 } else {
62 NOTREACHED(); 105 NOTREACHED();
63 } 106 }
64 } 107 }
65 } 108 }
66 109
110 // static
111 void NetworkChangeNotifierMac::ReachabilityCallback(
112 SCNetworkReachabilityRef target,
113 SCNetworkConnectionFlags flags,
114 void* notifier) {
115 NetworkChangeNotifierMac* notifier_mac =
116 static_cast<NetworkChangeNotifierMac*>(notifier);
117
118 DCHECK_EQ(notifier_mac->run_loop_.get(), CFRunLoopGetCurrent());
119
120 bool reachable = flags & kSCNetworkFlagsReachable;
121 bool connection_required = flags & kSCNetworkFlagsConnectionRequired;
122 bool old_reachability = notifier_mac->network_reachable_;
123 notifier_mac->network_reachable_ = reachable && !connection_required;
124 if (old_reachability != notifier_mac->network_reachable_)
125 notifier_mac->NotifyObserversOfOnlineStateChange();
126 }
127
67 } // namespace net 128 } // namespace net
OLDNEW
« no previous file with comments | « net/base/network_change_notifier_mac.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698