OLD | NEW |
---|---|
1 // Copyright 2017 The Chromium Authors. All rights reserved. | 1 // Copyright 2017 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 "chrome/browser/media/router/discovery/mdns/cast_media_sink_service.h" | 5 #include "chrome/browser/media/router/discovery/mdns/cast_media_sink_service.h" |
6 | 6 |
7 #include "base/memory/ptr_util.h" | 7 #include "base/memory/ptr_util.h" |
8 #include "chrome/browser/browser_process.h" | 8 #include "chrome/browser/browser_process.h" |
9 #include "chrome/common/media_router/discovery/media_sink_internal.h" | 9 #include "chrome/common/media_router/discovery/media_sink_internal.h" |
10 #include "components/cast_channel/cast_socket_service.h" | 10 #include "components/cast_channel/cast_socket_service.h" |
(...skipping 10 matching lines...) Expand all Loading... | |
21 enum ErrorType { | 21 enum ErrorType { |
22 NONE, | 22 NONE, |
23 NOT_CAST_DEVICE, | 23 NOT_CAST_DEVICE, |
24 MISSING_ID, | 24 MISSING_ID, |
25 MISSING_FRIENDLY_NAME, | 25 MISSING_FRIENDLY_NAME, |
26 MISSING_OR_INVALID_IP_ADDRESS, | 26 MISSING_OR_INVALID_IP_ADDRESS, |
27 MISSING_OR_INVALID_PORT, | 27 MISSING_OR_INVALID_PORT, |
28 }; | 28 }; |
29 | 29 |
30 ErrorType CreateCastMediaSink(const media_router::DnsSdService& service, | 30 ErrorType CreateCastMediaSink(const media_router::DnsSdService& service, |
31 int channel_id, | |
32 bool audio_only, | |
33 media_router::MediaSinkInternal* cast_sink) { | 31 media_router::MediaSinkInternal* cast_sink) { |
34 DCHECK(cast_sink); | 32 DCHECK(cast_sink); |
35 if (service.service_name.find( | 33 if (service.service_name.find( |
36 media_router::CastMediaSinkService::kCastServiceType) == | 34 media_router::CastMediaSinkService::kCastServiceType) == |
37 std::string::npos) | 35 std::string::npos) |
38 return ErrorType::NOT_CAST_DEVICE; | 36 return ErrorType::NOT_CAST_DEVICE; |
39 | 37 |
40 net::IPAddress ip_address; | 38 net::IPAddress ip_address; |
41 if (!ip_address.AssignFromIPLiteral(service.ip_address)) | 39 if (!ip_address.AssignFromIPLiteral(service.ip_address)) |
42 return ErrorType::MISSING_OR_INVALID_IP_ADDRESS; | 40 return ErrorType::MISSING_OR_INVALID_IP_ADDRESS; |
43 | 41 |
42 net::HostPortPair host_port_pair = | |
mark a. foltz
2017/07/11 23:52:51
It looks like the local_discovery::ServiceDescript
| |
43 net::HostPortPair::FromString(service.service_host_port); | |
44 int port = host_port_pair.port(); | |
45 | |
44 std::map<std::string, std::string> service_data; | 46 std::map<std::string, std::string> service_data; |
45 for (const auto& item : service.service_data) { | 47 for (const auto& item : service.service_data) { |
46 // |item| format should be "id=xxxxxx", etc. | 48 // |item| format should be "id=xxxxxx", etc. |
47 size_t split_idx = item.find('='); | 49 size_t split_idx = item.find('='); |
48 if (split_idx == std::string::npos) | 50 if (split_idx == std::string::npos) |
49 continue; | 51 continue; |
50 | 52 |
51 std::string key = item.substr(0, split_idx); | 53 std::string key = item.substr(0, split_idx); |
52 std::string val = | 54 std::string val = |
53 split_idx < item.length() ? item.substr(split_idx + 1) : ""; | 55 split_idx < item.length() ? item.substr(split_idx + 1) : ""; |
54 service_data[key] = val; | 56 service_data[key] = val; |
55 } | 57 } |
56 | 58 |
57 // When use this "sink" within browser, please note it will have a different | 59 // When use this "sink" within browser, please note it will have a different |
58 // ID when it is sent to the extension, because it derives a different sink ID | 60 // ID when it is sent to the extension, because it derives a different sink ID |
59 // using the given sink ID. | 61 // using the given sink ID. |
60 std::string unique_id = service_data["id"]; | 62 std::string unique_id = service_data["id"]; |
61 if (unique_id.empty()) | 63 if (unique_id.empty()) |
62 return ErrorType::MISSING_ID; | 64 return ErrorType::MISSING_ID; |
63 std::string friendly_name = service_data["fn"]; | 65 std::string friendly_name = service_data["fn"]; |
64 if (friendly_name.empty()) | 66 if (friendly_name.empty()) |
65 return ErrorType::MISSING_FRIENDLY_NAME; | 67 return ErrorType::MISSING_FRIENDLY_NAME; |
66 media_router::MediaSink sink(unique_id, friendly_name, | 68 media_router::MediaSink sink(unique_id, friendly_name, |
67 media_router::MediaSink::IconType::CAST); | 69 media_router::MediaSink::IconType::CAST); |
68 | 70 |
69 media_router::CastSinkExtraData extra_data; | 71 media_router::CastSinkExtraData extra_data; |
70 extra_data.ip_address = ip_address; | 72 extra_data.ip_address = ip_address; |
71 extra_data.model_name = service_data["md"]; | 73 extra_data.model_name = service_data["md"]; |
72 extra_data.capabilities = cast_channel::CastDeviceCapability::AUDIO_OUT; | 74 extra_data.port = port; |
73 if (!audio_only) | 75 extra_data.discovered_by_dial = false; |
74 extra_data.capabilities |= cast_channel::CastDeviceCapability::VIDEO_OUT; | |
75 extra_data.cast_channel_id = channel_id; | |
76 | 76 |
77 cast_sink->set_sink(sink); | 77 cast_sink->set_sink(sink); |
78 cast_sink->set_cast_data(extra_data); | 78 cast_sink->set_cast_data(extra_data); |
79 | 79 |
80 return ErrorType::NONE; | 80 return ErrorType::NONE; |
81 } | 81 } |
82 | 82 |
83 static media_router::MediaSinkInternal CreateCastSinkFromDialSink( | |
84 const media_router::MediaSinkInternal& dial_sink) { | |
85 std::string unique_id = dial_sink.sink().id(); | |
86 std::string friendly_name = dial_sink.sink().name(); | |
87 media_router::MediaSink sink(unique_id, friendly_name, | |
88 media_router::MediaSink::IconType::CAST); | |
89 | |
90 media_router::CastSinkExtraData extra_data; | |
91 extra_data.ip_address = dial_sink.dial_data().ip_address; | |
92 extra_data.port = media_router::CastMediaSinkService::kCastControlPort; | |
93 extra_data.model_name = dial_sink.dial_data().model_name; | |
94 extra_data.discovered_by_dial = true; | |
95 | |
96 return media_router::MediaSinkInternal(sink, extra_data); | |
97 } | |
98 | |
83 } // namespace | 99 } // namespace |
84 | 100 |
85 namespace media_router { | 101 namespace media_router { |
86 | 102 |
87 // static | 103 // static |
88 const char CastMediaSinkService::kCastServiceType[] = "_googlecast._tcp.local"; | 104 const char CastMediaSinkService::kCastServiceType[] = "_googlecast._tcp.local"; |
89 | 105 |
106 // static | |
107 const int CastMediaSinkService::kCastControlPort = 8009; | |
108 | |
90 CastMediaSinkService::CastMediaSinkService( | 109 CastMediaSinkService::CastMediaSinkService( |
91 const OnSinksDiscoveredCallback& callback, | 110 const OnSinksDiscoveredCallback& callback, |
92 content::BrowserContext* browser_context) | 111 content::BrowserContext* browser_context) |
93 : MediaSinkServiceBase(callback) { | 112 : MediaSinkServiceBase(callback) { |
94 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | 113 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
95 cast_socket_service_ = cast_channel::CastSocketServiceFactory::GetInstance() | 114 cast_socket_service_ = cast_channel::CastSocketServiceFactory::GetInstance() |
96 ->GetForBrowserContext(browser_context); | 115 ->GetForBrowserContext(browser_context); |
97 DCHECK(cast_socket_service_); | 116 DCHECK(cast_socket_service_); |
98 } | 117 } |
99 | 118 |
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
131 } | 150 } |
132 | 151 |
133 void CastMediaSinkService::SetDnsSdRegistryForTest(DnsSdRegistry* registry) { | 152 void CastMediaSinkService::SetDnsSdRegistryForTest(DnsSdRegistry* registry) { |
134 DCHECK(!dns_sd_registry_); | 153 DCHECK(!dns_sd_registry_); |
135 dns_sd_registry_ = registry; | 154 dns_sd_registry_ = registry; |
136 dns_sd_registry_->AddObserver(this); | 155 dns_sd_registry_->AddObserver(this); |
137 dns_sd_registry_->RegisterDnsSdListener(kCastServiceType); | 156 dns_sd_registry_->RegisterDnsSdListener(kCastServiceType); |
138 MediaSinkServiceBase::StartTimer(); | 157 MediaSinkServiceBase::StartTimer(); |
139 } | 158 } |
140 | 159 |
160 void CastMediaSinkService::OnFetchCompleted() { | |
161 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | |
162 | |
163 auto task_runner = content::BrowserThread::GetTaskRunnerForThread( | |
164 content::BrowserThread::IO); | |
imcheng
2017/07/12 00:51:36
It might be easier to migrate this code to use Tas
| |
165 base::PostTaskAndReplyWithResult( | |
166 task_runner.get(), FROM_HERE, | |
167 base::BindOnce(&CastMediaSinkService::GetCastSinksOnIOThread, this), | |
imcheng
2017/07/12 00:51:36
Why do we need a thread hop now, but did not befor
| |
168 base::BindOnce(&CastMediaSinkService::OnFetchCompletedOnUIThread, this)); | |
169 } | |
170 | |
171 std::set<MediaSinkInternal> CastMediaSinkService::GetCastSinksOnIOThread() { | |
172 DCHECK_CURRENTLY_ON(content::BrowserThread::IO); | |
173 | |
174 std::set<MediaSinkInternal> sinks; | |
175 // Copy cast sink from mDNS service to |current_sinks_|. | |
176 for (const auto& sink_it : current_sinks_by_mdns_map_) { | |
177 DVLOG(2) << "Discovered by mdns [name]: " << sink_it.second.sink().name(); | |
178 sinks.insert(sink_it.second); | |
179 } | |
180 | |
181 // Copy cast sink from DIAL discovery to |current_sinks_|. | |
182 for (const auto& sink_it : current_sinks_by_dial_map_) { | |
183 DVLOG(2) << "Discovered by dial [name]: " << sink_it.second.sink().name(); | |
184 if (!base::ContainsKey(current_sinks_by_mdns_map_, sink_it.first)) | |
185 sinks.insert(sink_it.second); | |
186 } | |
187 return sinks; | |
188 } | |
189 | |
190 void CastMediaSinkService::OnFetchCompletedOnUIThread( | |
191 std::set<MediaSinkInternal> cast_sinks) { | |
192 MediaSinkServiceBase::current_sinks_ = cast_sinks; | |
193 MediaSinkServiceBase::OnFetchCompleted(); | |
194 } | |
195 | |
141 void CastMediaSinkService::OnDnsSdEvent( | 196 void CastMediaSinkService::OnDnsSdEvent( |
142 const std::string& service_type, | 197 const std::string& service_type, |
143 const DnsSdRegistry::DnsSdServiceList& services) { | 198 const DnsSdRegistry::DnsSdServiceList& services) { |
144 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | 199 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
145 DVLOG(2) << "CastMediaSinkService::OnDnsSdEvent found " << services.size() | 200 DVLOG(2) << "CastMediaSinkService::OnDnsSdEvent found " << services.size() |
146 << " services"; | 201 << " services"; |
147 | 202 |
148 current_sinks_.clear(); | 203 std::vector<MediaSinkInternal> cast_sinks; |
149 current_services_ = services; | |
150 | |
151 for (const auto& service : services) { | 204 for (const auto& service : services) { |
152 net::IPAddress ip_address; | 205 // Create Cast sink from mDNS service description. |
153 if (!ip_address.AssignFromIPLiteral(service.ip_address)) { | 206 MediaSinkInternal cast_sink; |
154 DVLOG(2) << "Invalid ip_addresss: " << service.ip_address; | 207 ErrorType error = CreateCastMediaSink(service, &cast_sink); |
208 if (error != ErrorType::NONE) { | |
209 DVLOG(2) << "Fail to create Cast device [error]: " << error; | |
155 continue; | 210 continue; |
156 } | 211 } |
157 net::HostPortPair host_port_pair = | |
158 net::HostPortPair::FromString(service.service_host_port); | |
159 | 212 |
160 content::BrowserThread::PostTask( | 213 cast_sinks.push_back(cast_sink); |
imcheng
2017/07/12 00:51:36
std::move(cast_sink)
| |
161 content::BrowserThread::IO, FROM_HERE, | |
162 base::Bind(&CastMediaSinkService::OpenChannelOnIOThread, this, service, | |
163 net::IPEndPoint(ip_address, host_port_pair.port()))); | |
164 } | 214 } |
165 | 215 |
216 content::BrowserThread::PostTask( | |
217 content::BrowserThread::IO, FROM_HERE, | |
218 base::Bind(&CastMediaSinkService::OpenChannelsOnIOThread, this, | |
219 std::move(cast_sinks))); | |
220 | |
166 MediaSinkServiceBase::RestartTimer(); | 221 MediaSinkServiceBase::RestartTimer(); |
167 } | 222 } |
168 | 223 |
224 void CastMediaSinkService::OpenChannelsOnIOThread( | |
225 std::vector<MediaSinkInternal> cast_sinks) { | |
226 DCHECK_CURRENTLY_ON(content::BrowserThread::IO); | |
227 | |
228 current_sinks_by_mdns_map_.clear(); | |
229 current_service_ip_endpoints_.clear(); | |
230 | |
231 for (const auto& cast_sink : cast_sinks) { | |
imcheng
2017/07/12 00:51:36
No const, so you can std::move in L236.
| |
232 net::IPEndPoint ip_endpoint(cast_sink.cast_data().ip_address, | |
233 cast_sink.cast_data().port); | |
234 current_service_ip_endpoints_.insert(ip_endpoint); | |
235 | |
236 OpenChannelOnIOThread(ip_endpoint, cast_sink); | |
237 } | |
238 } | |
239 | |
240 void CastMediaSinkService::OnDialSinkAdded(const MediaSinkInternal& sink) { | |
241 DCHECK_CURRENTLY_ON(content::BrowserThread::IO); | |
242 | |
243 auto ip_address = sink.dial_data().ip_address; | |
244 net::IPEndPoint ip_endpoint(ip_address, kCastControlPort); | |
245 | |
246 if (base::ContainsKey(current_service_ip_endpoints_, ip_endpoint)) { | |
247 DVLOG(2) << "Sink discovered by mDNS, skip adding [name]: " | |
248 << sink.sink().name(); | |
249 return; | |
250 } | |
251 | |
252 auto cast_sink = CreateCastSinkFromDialSink(sink); | |
253 OpenChannelOnIOThread(ip_endpoint, cast_sink); | |
imcheng
2017/07/12 00:51:36
Inline CreateCastSinkFromDialSink here.
| |
254 } | |
255 | |
256 void CastMediaSinkService::OnDialSinksRemoved() { | |
257 DCHECK_CURRENTLY_ON(content::BrowserThread::IO); | |
258 | |
259 current_sinks_by_dial_map_.clear(); | |
260 } | |
261 | |
169 void CastMediaSinkService::OpenChannelOnIOThread( | 262 void CastMediaSinkService::OpenChannelOnIOThread( |
170 const DnsSdService& service, | 263 const net::IPEndPoint& ip_endpoint, |
171 const net::IPEndPoint& ip_endpoint) { | 264 const MediaSinkInternal& cast_sink) { |
imcheng
2017/07/12 00:51:36
pass by value here, since it will be moved.
| |
172 auto* observer = cast_socket_service_->GetObserver(kObserverId); | 265 auto* observer = cast_socket_service_->GetObserver(kObserverId); |
173 if (!observer) { | 266 if (!observer) { |
174 observer = new CastSocketObserver(); | 267 observer = new CastSocketObserver(); |
175 cast_socket_service_->AddObserver(kObserverId, base::WrapUnique(observer)); | 268 cast_socket_service_->AddObserver(kObserverId, base::WrapUnique(observer)); |
176 } | 269 } |
177 | 270 |
178 cast_socket_service_->OpenSocket( | 271 cast_socket_service_->OpenSocket( |
179 ip_endpoint, g_browser_process->net_log(), | 272 ip_endpoint, g_browser_process->net_log(), |
180 base::Bind(&CastMediaSinkService::OnChannelOpenedOnIOThread, this, | 273 base::Bind(&CastMediaSinkService::OnChannelOpenedOnIOThread, this, |
imcheng
2017/07/12 00:51:36
BindOnce?
| |
181 service), | 274 std::move(cast_sink)), |
182 observer); | 275 observer); |
183 } | 276 } |
184 | 277 |
185 void CastMediaSinkService::OnChannelOpenedOnIOThread( | 278 void CastMediaSinkService::OnChannelOpenedOnIOThread( |
186 const DnsSdService& service, | 279 MediaSinkInternal cast_sink, |
187 int channel_id, | 280 int channel_id, |
188 cast_channel::ChannelError channel_error) { | 281 cast_channel::ChannelError channel_error) { |
189 if (channel_error != cast_channel::ChannelError::NONE) { | 282 if (channel_error != cast_channel::ChannelError::NONE) { |
190 DVLOG(2) << "Fail to open channel " << service.ip_address << ": " | 283 DVLOG(2) << "Fail to open channel " |
191 << service.service_host_port | 284 << cast_sink.cast_data().ip_address.ToString() |
285 << " [name]: " << cast_sink.sink().name() | |
192 << " [ChannelError]: " << (int)channel_error; | 286 << " [ChannelError]: " << (int)channel_error; |
193 return; | 287 return; |
194 } | 288 } |
195 | 289 |
196 auto* socket = cast_socket_service_->GetSocket(channel_id); | 290 auto* socket = cast_socket_service_->GetSocket(channel_id); |
197 if (!socket) { | 291 if (!socket) { |
198 DVLOG(2) << "Fail to find socket with [channel_id]: " << channel_id; | 292 DVLOG(2) << "Fail to find socket with [channel_id]: " << channel_id; |
199 return; | 293 return; |
200 } | 294 } |
201 | 295 |
296 // Skip adding if IP address not found in current round of mDNS discovery. | |
297 net::IPEndPoint ip_endpoint(cast_sink.cast_data().ip_address, | |
298 cast_sink.cast_data().port); | |
299 if (!cast_sink.cast_data().discovered_by_dial) { | |
300 if (current_service_ip_endpoints_.find(ip_endpoint) == | |
301 current_service_ip_endpoints_.end()) { | |
302 DVLOG(2) << "Service data not found in current service data list..." | |
303 << ip_endpoint.ToString(); | |
304 return; | |
305 } | |
306 } | |
307 | |
308 media_router::CastSinkExtraData extra_data = cast_sink.cast_data(); | |
309 extra_data.capabilities = cast_channel::CastDeviceCapability::AUDIO_OUT; | |
310 if (!socket->audio_only()) | |
311 extra_data.capabilities |= cast_channel::CastDeviceCapability::VIDEO_OUT; | |
312 extra_data.cast_channel_id = channel_id; | |
313 | |
314 MediaSinkInternal updated_sink(cast_sink.sink(), extra_data); | |
315 DVLOG(2) << "Ading sink to current_sinks_ [name]: " | |
316 << updated_sink.sink().name(); | |
317 | |
318 // Add or update existing cast sink. | |
319 if (updated_sink.cast_data().discovered_by_dial) { | |
320 current_sinks_by_dial_map_[ip_endpoint] = updated_sink; | |
imcheng
2017/07/12 00:51:36
nit: I personally think what you had in PS3 (updat
| |
321 } else { | |
322 current_sinks_by_mdns_map_[ip_endpoint] = updated_sink; | |
323 } | |
324 | |
202 content::BrowserThread::PostTask( | 325 content::BrowserThread::PostTask( |
203 content::BrowserThread::UI, FROM_HERE, | 326 content::BrowserThread::UI, FROM_HERE, |
204 base::Bind(&CastMediaSinkService::OnChannelOpenedOnUIThread, this, | 327 base::Bind(&CastMediaSinkService::RestartTimer, this)); |
205 service, channel_id, socket->audio_only())); | |
206 } | |
207 | |
208 void CastMediaSinkService::OnChannelOpenedOnUIThread( | |
209 const DnsSdService& service, | |
210 int channel_id, | |
211 bool audio_only) { | |
212 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | |
213 MediaSinkInternal sink; | |
214 ErrorType error = CreateCastMediaSink(service, channel_id, audio_only, &sink); | |
215 if (error != ErrorType::NONE) { | |
216 DVLOG(2) << "Fail to create Cast device [error]: " << error; | |
217 return; | |
218 } | |
219 | |
220 if (!base::ContainsValue(current_services_, service)) { | |
221 DVLOG(2) << "Service data not found in current service data list..."; | |
222 return; | |
223 } | |
224 | |
225 DVLOG(2) << "Ading sink to current_sinks_ [id]: " << sink.sink().id(); | |
226 current_sinks_.insert(sink); | |
227 MediaSinkServiceBase::RestartTimer(); | |
228 } | 328 } |
229 | 329 |
230 CastMediaSinkService::CastSocketObserver::CastSocketObserver() {} | 330 CastMediaSinkService::CastSocketObserver::CastSocketObserver() {} |
231 CastMediaSinkService::CastSocketObserver::~CastSocketObserver() {} | 331 CastMediaSinkService::CastSocketObserver::~CastSocketObserver() {} |
232 | 332 |
233 void CastMediaSinkService::CastSocketObserver::OnError( | 333 void CastMediaSinkService::CastSocketObserver::OnError( |
234 const cast_channel::CastSocket& socket, | 334 const cast_channel::CastSocket& socket, |
235 cast_channel::ChannelError error_state) { | 335 cast_channel::ChannelError error_state) { |
236 DVLOG(1) << "OnError [ip_endpoint]: " << socket.ip_endpoint().ToString() | 336 DVLOG(1) << "OnError [ip_endpoint]: " << socket.ip_endpoint().ToString() |
237 << " [error_state]: " | 337 << " [error_state]: " |
238 << cast_channel::ChannelErrorToString(error_state); | 338 << cast_channel::ChannelErrorToString(error_state); |
239 } | 339 } |
240 | 340 |
241 void CastMediaSinkService::CastSocketObserver::OnMessage( | 341 void CastMediaSinkService::CastSocketObserver::OnMessage( |
242 const cast_channel::CastSocket& socket, | 342 const cast_channel::CastSocket& socket, |
243 const cast_channel::CastMessage& message) {} | 343 const cast_channel::CastMessage& message) {} |
244 | 344 |
245 } // namespace media_router | 345 } // namespace media_router |
OLD | NEW |