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 "content/browser/geolocation/geolocation_provider.h" | 5 #include "content/browser/geolocation/geolocation_provider.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "base/bind_helpers.h" | |
9 #include "base/memory/singleton.h" | 8 #include "base/memory/singleton.h" |
10 #include "base/threading/thread_restrictions.h" | |
11 #include "content/browser/geolocation/location_arbitrator.h" | 9 #include "content/browser/geolocation/location_arbitrator.h" |
| 10 #include "content/public/browser/browser_thread.h" |
12 | 11 |
13 GeolocationProvider* GeolocationProvider::GetInstance() { | 12 GeolocationProvider* GeolocationProvider::GetInstance() { |
| 13 DCHECK(OnIOThread()); |
14 return Singleton<GeolocationProvider>::get(); | 14 return Singleton<GeolocationProvider>::get(); |
15 } | 15 } |
16 | 16 |
17 GeolocationProvider::GeolocationProvider() | 17 GeolocationProvider::GeolocationProvider() |
18 : base::Thread("Geolocation"), | 18 : base::Thread("Geolocation"), |
19 client_loop_(base::MessageLoopProxy::current()), | |
20 is_permission_granted_(false), | 19 is_permission_granted_(false), |
21 arbitrator_(NULL) { | 20 arbitrator_(NULL) { |
| 21 DCHECK(OnIOThread()); |
22 } | 22 } |
23 | 23 |
24 GeolocationProvider::~GeolocationProvider() { | 24 GeolocationProvider::~GeolocationProvider() { |
25 DCHECK(observers_.empty()); // observers must unregister. | 25 DCHECK(observers_.empty()); // observers must unregister. |
26 Stop(); | 26 Stop(); |
27 DCHECK(!arbitrator_); | 27 DCHECK(!arbitrator_); |
28 } | 28 } |
29 | 29 |
30 void GeolocationProvider::AddObserver(GeolocationObserver* observer, | 30 void GeolocationProvider::AddObserver(GeolocationObserver* observer, |
31 const GeolocationObserverOptions& update_options) { | 31 const GeolocationObserverOptions& update_options) { |
32 DCHECK(OnClientThread()); | 32 DCHECK(OnIOThread()); |
33 observers_[observer] = update_options; | 33 observers_[observer] = update_options; |
34 OnObserversChanged(); | 34 OnObserversChanged(); |
35 if (position_.IsInitialized()) | 35 if (position_.IsInitialized()) |
36 observer->OnLocationUpdate(position_); | 36 observer->OnLocationUpdate(position_); |
37 } | 37 } |
38 | 38 |
39 bool GeolocationProvider::RemoveObserver(GeolocationObserver* observer) { | 39 bool GeolocationProvider::RemoveObserver(GeolocationObserver* observer) { |
40 DCHECK(OnClientThread()); | 40 DCHECK(OnIOThread()); |
41 size_t remove = observers_.erase(observer); | 41 size_t remove = observers_.erase(observer); |
42 OnObserversChanged(); | 42 OnObserversChanged(); |
43 return remove > 0; | 43 return remove > 0; |
44 } | 44 } |
45 | 45 |
46 void GeolocationProvider::OnObserversChanged() { | 46 void GeolocationProvider::OnObserversChanged() { |
47 DCHECK(OnClientThread()); | 47 DCHECK(OnIOThread()); |
48 base::Closure task; | 48 base::Closure task; |
49 if (observers_.empty()) { | 49 if (observers_.empty()) { |
50 DCHECK(IsRunning()); | 50 DCHECK(IsRunning()); |
51 task = base::Bind(&GeolocationProvider::StopProviders, | 51 task = base::Bind(&GeolocationProvider::StopProviders, |
52 base::Unretained(this)); | 52 base::Unretained(this)); |
53 } else { | 53 } else { |
54 if (!IsRunning()) { | 54 if (!IsRunning()) { |
55 Start(); | 55 Start(); |
56 if (HasPermissionBeenGranted()) | 56 if (HasPermissionBeenGranted()) |
57 InformProvidersPermissionGranted(); | 57 InformProvidersPermissionGranted(); |
58 } | 58 } |
59 | 59 |
60 // The high accuracy requirement may have changed. | 60 // The high accuracy requirement may have changed. |
61 task = base::Bind(&GeolocationProvider::StartProviders, | 61 task = base::Bind(&GeolocationProvider::StartProviders, |
62 base::Unretained(this), | 62 base::Unretained(this), |
63 GeolocationObserverOptions::Collapse(observers_)); | 63 GeolocationObserverOptions::Collapse(observers_)); |
64 } | 64 } |
65 | 65 |
66 message_loop()->PostTask(FROM_HERE, task); | 66 message_loop()->PostTask(FROM_HERE, task); |
67 } | 67 } |
68 | 68 |
69 void GeolocationProvider::NotifyObservers(const Geoposition& position) { | 69 void GeolocationProvider::NotifyObservers(const Geoposition& position) { |
70 DCHECK(OnClientThread()); | 70 DCHECK(OnIOThread()); |
71 DCHECK(position.IsInitialized()); | 71 DCHECK(position.IsInitialized()); |
72 position_ = position; | 72 position_ = position; |
73 ObserverMap::const_iterator it = observers_.begin(); | 73 ObserverMap::const_iterator it = observers_.begin(); |
74 while (it != observers_.end()) { | 74 while (it != observers_.end()) { |
75 // Advance iterator before callback to guard against synchronous unregister. | 75 // Advance iterator before callback to guard against synchronous unregister. |
76 GeolocationObserver* observer = it->first; | 76 GeolocationObserver* observer = it->first; |
77 ++it; | 77 ++it; |
78 observer->OnLocationUpdate(position_); | 78 observer->OnLocationUpdate(position_); |
79 } | 79 } |
80 } | 80 } |
81 | 81 |
82 void GeolocationProvider::StartProviders( | 82 void GeolocationProvider::StartProviders( |
83 const GeolocationObserverOptions& options) { | 83 const GeolocationObserverOptions& options) { |
84 DCHECK(OnGeolocationThread()); | 84 DCHECK(OnGeolocationThread()); |
85 DCHECK(arbitrator_); | 85 DCHECK(arbitrator_); |
86 arbitrator_->StartProviders(options); | 86 arbitrator_->StartProviders(options); |
87 } | 87 } |
88 | 88 |
89 void GeolocationProvider::StopProviders() { | 89 void GeolocationProvider::StopProviders() { |
90 DCHECK(OnGeolocationThread()); | 90 DCHECK(OnGeolocationThread()); |
91 DCHECK(arbitrator_); | 91 DCHECK(arbitrator_); |
92 arbitrator_->StopProviders(); | 92 arbitrator_->StopProviders(); |
93 } | 93 } |
94 | 94 |
95 void GeolocationProvider::OnPermissionGranted() { | 95 void GeolocationProvider::OnPermissionGranted() { |
96 DCHECK(OnClientThread()); | 96 DCHECK(OnIOThread()); |
97 is_permission_granted_ = true; | 97 is_permission_granted_ = true; |
98 if (IsRunning()) | 98 if (IsRunning()) |
99 InformProvidersPermissionGranted(); | 99 InformProvidersPermissionGranted(); |
100 } | 100 } |
101 | 101 |
102 void GeolocationProvider::InformProvidersPermissionGranted() { | 102 void GeolocationProvider::InformProvidersPermissionGranted() { |
103 DCHECK(IsRunning()); | 103 DCHECK(IsRunning()); |
104 if (!OnGeolocationThread()) { | 104 if (!OnGeolocationThread()) { |
105 message_loop()->PostTask( | 105 message_loop()->PostTask( |
106 FROM_HERE, | 106 FROM_HERE, |
(...skipping 13 matching lines...) Expand all Loading... |
120 } | 120 } |
121 | 121 |
122 void GeolocationProvider::CleanUp() { | 122 void GeolocationProvider::CleanUp() { |
123 DCHECK(OnGeolocationThread()); | 123 DCHECK(OnGeolocationThread()); |
124 delete arbitrator_; | 124 delete arbitrator_; |
125 arbitrator_ = NULL; | 125 arbitrator_ = NULL; |
126 } | 126 } |
127 | 127 |
128 void GeolocationProvider::OnLocationUpdate(const Geoposition& position) { | 128 void GeolocationProvider::OnLocationUpdate(const Geoposition& position) { |
129 DCHECK(OnGeolocationThread()); | 129 DCHECK(OnGeolocationThread()); |
130 client_loop_->PostTask( | 130 content::BrowserThread::PostTask( |
| 131 content::BrowserThread::IO, |
131 FROM_HERE, | 132 FROM_HERE, |
132 base::Bind(&GeolocationProvider::NotifyObservers, | 133 base::Bind(&GeolocationProvider::NotifyObservers, |
133 base::Unretained(this), position)); | 134 base::Unretained(this), position)); |
134 } | 135 } |
135 | 136 |
136 bool GeolocationProvider::HasPermissionBeenGranted() const { | 137 bool GeolocationProvider::HasPermissionBeenGranted() const { |
137 DCHECK(OnClientThread()); | 138 DCHECK(OnIOThread()); |
138 return is_permission_granted_; | 139 return is_permission_granted_; |
139 } | 140 } |
140 | 141 |
141 bool GeolocationProvider::OnClientThread() const { | 142 // static |
142 return client_loop_->BelongsToCurrentThread(); | 143 bool GeolocationProvider::OnIOThread() { |
| 144 return content::BrowserThread::CurrentlyOn(content::BrowserThread::IO); |
143 } | 145 } |
144 | 146 |
145 bool GeolocationProvider::OnGeolocationThread() const { | 147 bool GeolocationProvider::OnGeolocationThread() const { |
146 return MessageLoop::current() == message_loop(); | 148 return MessageLoop::current() == message_loop(); |
147 } | 149 } |
OLD | NEW |