Chromium Code Reviews| 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 = | |
| 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 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 MediaSinkServiceBase::RemoveSinks(); | |
| 164 | |
| 165 // Copy cast sink from mDNS service to |current_sinks_|. | |
| 166 for (const auto& sink_it : current_sinks_by_mdns_map_) | |
| 167 MediaSinkServiceBase::AddSink(sink_it.second); | |
| 168 | |
| 169 // Copy cast sink from DIAL discovery to |current_sinks_|. | |
| 170 for (const auto& sink_it : current_sinks_by_dial_map_) { | |
| 171 if (!base::ContainsKey(current_sinks_by_mdns_map_, sink_it.first)) | |
| 172 MediaSinkServiceBase::AddSink(sink_it.second); | |
| 173 } | |
| 174 MediaSinkServiceBase::OnFetchCompleted(); | |
| 175 } | |
| 176 | |
| 141 void CastMediaSinkService::OnDnsSdEvent( | 177 void CastMediaSinkService::OnDnsSdEvent( |
| 142 const std::string& service_type, | 178 const std::string& service_type, |
| 143 const DnsSdRegistry::DnsSdServiceList& services) { | 179 const DnsSdRegistry::DnsSdServiceList& services) { |
| 144 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | 180 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
| 145 DVLOG(2) << "CastMediaSinkService::OnDnsSdEvent found " << services.size() | 181 DVLOG(2) << "CastMediaSinkService::OnDnsSdEvent found " << services.size() |
| 146 << " services"; | 182 << " services"; |
| 147 | 183 |
| 148 current_sinks_.clear(); | 184 current_sinks_by_mdns_map_.clear(); |
| 149 current_services_ = services; | 185 current_service_ip_endpoints_.clear(); |
| 150 | 186 |
| 151 for (const auto& service : services) { | 187 for (const auto& service : services) { |
| 152 net::IPAddress ip_address; | 188 // Create Cast sink from mDNS service description. |
| 153 if (!ip_address.AssignFromIPLiteral(service.ip_address)) { | 189 MediaSinkInternal cast_sink; |
| 154 DVLOG(2) << "Invalid ip_addresss: " << service.ip_address; | 190 ErrorType error = CreateCastMediaSink(service, &cast_sink); |
| 191 if (error != ErrorType::NONE) { | |
| 192 DVLOG(2) << "Fail to create Cast device [error]: " << error; | |
| 155 continue; | 193 continue; |
| 156 } | 194 } |
| 157 net::HostPortPair host_port_pair = | |
| 158 net::HostPortPair::FromString(service.service_host_port); | |
| 159 | 195 |
| 196 net::IPEndPoint ip_endpoint(cast_sink.cast_data().ip_address, | |
| 197 cast_sink.cast_data().port); | |
| 198 current_service_ip_endpoints_.insert(ip_endpoint); | |
| 199 | |
| 200 // Open Cast socket on the IO thread. | |
| 160 content::BrowserThread::PostTask( | 201 content::BrowserThread::PostTask( |
| 161 content::BrowserThread::IO, FROM_HERE, | 202 content::BrowserThread::IO, FROM_HERE, |
| 162 base::Bind(&CastMediaSinkService::OpenChannelOnIOThread, this, service, | 203 base::Bind(&CastMediaSinkService::OpenChannelOnIOThread, this, |
| 163 net::IPEndPoint(ip_address, host_port_pair.port()))); | 204 ip_endpoint, |
| 205 base::Bind(&CastMediaSinkService::OnChannelOpenedOnIOThread, | |
| 206 this, cast_sink))); | |
| 164 } | 207 } |
| 165 | 208 |
| 166 MediaSinkServiceBase::RestartTimer(); | 209 MediaSinkServiceBase::RestartTimer(); |
| 167 } | 210 } |
| 168 | 211 |
| 212 void CastMediaSinkService::OnMediaSinkAdded(const MediaSinkInternal& sink) { | |
| 213 auto ip_address = sink.dial_data().ip_address; | |
| 214 net::IPEndPoint ip_endpoint(ip_address, kCastControlPort); | |
| 215 | |
| 216 if (base::ContainsKey(current_service_ip_endpoints_, ip_endpoint)) { | |
| 217 DVLOG(2) << "Sink discovered by mDNS, skip adding [name]: " | |
| 218 << sink.sink().name(); | |
| 219 return; | |
| 220 } | |
| 221 | |
| 222 auto cast_sink = CreateCastSinkFromDialSink(sink); | |
| 223 OpenChannelOnIOThread( | |
| 224 ip_endpoint, base::Bind(&CastMediaSinkService::OnChannelOpenedOnIOThread, | |
| 225 this, cast_sink)); | |
|
imcheng
2017/07/06 22:53:30
We can enable moves for MediaSinkInternal to make
zhaobin
2017/07/10 20:21:15
Done.
| |
| 226 } | |
| 227 | |
| 228 void CastMediaSinkService::OnMediaSinksRemoved() { | |
| 229 current_sinks_by_dial_map_.clear(); | |
| 230 } | |
| 231 | |
| 169 void CastMediaSinkService::OpenChannelOnIOThread( | 232 void CastMediaSinkService::OpenChannelOnIOThread( |
| 170 const DnsSdService& service, | 233 const net::IPEndPoint& ip_endpoint, |
| 171 const net::IPEndPoint& ip_endpoint) { | 234 const cast_channel::CastSocket::OnOpenCallback& on_open_callback) { |
|
imcheng
2017/07/06 22:53:30
Does this need to take a callback as input, or can
zhaobin
2017/07/10 20:21:15
Done.
| |
| 172 auto* observer = cast_socket_service_->GetObserver(kObserverId); | 235 auto* observer = cast_socket_service_->GetObserver(kObserverId); |
| 173 if (!observer) { | 236 if (!observer) { |
| 174 observer = new CastSocketObserver(); | 237 observer = new CastSocketObserver(); |
| 175 cast_socket_service_->AddObserver(kObserverId, base::WrapUnique(observer)); | 238 cast_socket_service_->AddObserver(kObserverId, base::WrapUnique(observer)); |
| 176 } | 239 } |
| 177 | 240 |
| 178 cast_socket_service_->OpenSocket( | 241 cast_socket_service_->OpenSocket(ip_endpoint, g_browser_process->net_log(), |
| 179 ip_endpoint, g_browser_process->net_log(), | 242 on_open_callback, observer); |
| 180 base::Bind(&CastMediaSinkService::OnChannelOpenedOnIOThread, this, | |
| 181 service), | |
| 182 observer); | |
| 183 } | 243 } |
| 184 | 244 |
| 185 void CastMediaSinkService::OnChannelOpenedOnIOThread( | 245 void CastMediaSinkService::OnChannelOpenedOnIOThread( |
| 186 const DnsSdService& service, | 246 const MediaSinkInternal& cast_sink, |
| 187 int channel_id, | 247 int channel_id, |
| 188 cast_channel::ChannelError channel_error) { | 248 cast_channel::ChannelError channel_error) { |
| 189 if (channel_error != cast_channel::ChannelError::NONE) { | 249 if (channel_error != cast_channel::ChannelError::NONE) { |
| 190 DVLOG(2) << "Fail to open channel " << service.ip_address << ": " | 250 DVLOG(2) << "Fail to open channel " |
| 191 << service.service_host_port | 251 << cast_sink.cast_data().ip_address.ToString() |
| 192 << " [ChannelError]: " << (int)channel_error; | 252 << " [ChannelError]: " << (int)channel_error; |
| 193 return; | 253 return; |
| 194 } | 254 } |
| 195 | 255 |
| 196 auto* socket = cast_socket_service_->GetSocket(channel_id); | 256 auto* socket = cast_socket_service_->GetSocket(channel_id); |
| 197 if (!socket) { | 257 if (!socket) { |
| 198 DVLOG(2) << "Fail to find socket with [channel_id]: " << channel_id; | 258 DVLOG(2) << "Fail to find socket with [channel_id]: " << channel_id; |
| 199 return; | 259 return; |
| 200 } | 260 } |
| 201 | 261 |
| 262 media_router::CastSinkExtraData extra_data = cast_sink.cast_data(); | |
| 263 extra_data.capabilities = cast_channel::CastDeviceCapability::AUDIO_OUT; | |
| 264 if (!socket->audio_only()) | |
| 265 extra_data.capabilities |= cast_channel::CastDeviceCapability::VIDEO_OUT; | |
| 266 extra_data.cast_channel_id = channel_id; | |
| 267 | |
| 202 content::BrowserThread::PostTask( | 268 content::BrowserThread::PostTask( |
| 203 content::BrowserThread::UI, FROM_HERE, | 269 content::BrowserThread::UI, FROM_HERE, |
| 204 base::Bind(&CastMediaSinkService::OnChannelOpenedOnUIThread, this, | 270 base::Bind(&CastMediaSinkService::OnChannelOpenedOnUIThread, this, |
| 205 service, channel_id, socket->audio_only())); | 271 MediaSinkInternal(cast_sink.sink(), extra_data))); |
| 206 } | 272 } |
| 207 | 273 |
| 208 void CastMediaSinkService::OnChannelOpenedOnUIThread( | 274 void CastMediaSinkService::OnChannelOpenedOnUIThread( |
| 209 const DnsSdService& service, | 275 const MediaSinkInternal& cast_sink) { |
| 210 int channel_id, | |
| 211 bool audio_only) { | |
| 212 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | 276 DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |
| 213 MediaSinkInternal sink; | 277 auto ip_endpoint = net::IPEndPoint(cast_sink.cast_data().ip_address, |
| 214 ErrorType error = CreateCastMediaSink(service, channel_id, audio_only, &sink); | 278 cast_sink.cast_data().port); |
| 215 if (error != ErrorType::NONE) { | 279 |
| 216 DVLOG(2) << "Fail to create Cast device [error]: " << error; | 280 // Skip adding if IP address not found in current round of mDNS discovery. |
| 217 return; | 281 if (!cast_sink.cast_data().discovered_by_dial) { |
| 282 if (!base::ContainsValue(current_service_ip_endpoints_, ip_endpoint)) { | |
| 283 DVLOG(2) << "Service data not found in current service data list..."; | |
| 284 return; | |
| 285 } | |
| 218 } | 286 } |
| 219 | 287 |
| 220 if (!base::ContainsValue(current_services_, service)) { | 288 DVLOG(2) << "Ading sink to current_sinks_ [name]: " |
| 221 DVLOG(2) << "Service data not found in current service data list..."; | 289 << cast_sink.sink().name(); |
| 222 return; | 290 // Add or update existing cast sink. |
| 291 if (cast_sink.cast_data().discovered_by_dial) { | |
| 292 current_sinks_by_dial_map_[ip_endpoint] = cast_sink; | |
| 293 } else { | |
| 294 current_sinks_by_mdns_map_[ip_endpoint] = cast_sink; | |
| 223 } | 295 } |
| 224 | |
| 225 DVLOG(2) << "Ading sink to current_sinks_ [id]: " << sink.sink().id(); | |
| 226 current_sinks_.insert(sink); | |
| 227 MediaSinkServiceBase::RestartTimer(); | 296 MediaSinkServiceBase::RestartTimer(); |
| 228 } | 297 } |
| 229 | 298 |
| 230 CastMediaSinkService::CastSocketObserver::CastSocketObserver() {} | 299 CastMediaSinkService::CastSocketObserver::CastSocketObserver() {} |
| 231 CastMediaSinkService::CastSocketObserver::~CastSocketObserver() {} | 300 CastMediaSinkService::CastSocketObserver::~CastSocketObserver() {} |
| 232 | 301 |
| 233 void CastMediaSinkService::CastSocketObserver::OnError( | 302 void CastMediaSinkService::CastSocketObserver::OnError( |
| 234 const cast_channel::CastSocket& socket, | 303 const cast_channel::CastSocket& socket, |
| 235 cast_channel::ChannelError error_state) { | 304 cast_channel::ChannelError error_state) { |
| 236 DVLOG(1) << "OnError [ip_endpoint]: " << socket.ip_endpoint().ToString() | 305 DVLOG(1) << "OnError [ip_endpoint]: " << socket.ip_endpoint().ToString() |
| 237 << " [error_state]: " | 306 << " [error_state]: " |
| 238 << cast_channel::ChannelErrorToString(error_state); | 307 << cast_channel::ChannelErrorToString(error_state); |
| 239 } | 308 } |
| 240 | 309 |
| 241 void CastMediaSinkService::CastSocketObserver::OnMessage( | 310 void CastMediaSinkService::CastSocketObserver::OnMessage( |
| 242 const cast_channel::CastSocket& socket, | 311 const cast_channel::CastSocket& socket, |
| 243 const cast_channel::CastMessage& message) {} | 312 const cast_channel::CastMessage& message) {} |
| 244 | 313 |
| 245 } // namespace media_router | 314 } // namespace media_router |
| OLD | NEW |