OLD | NEW |
---|---|
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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/webrtc_logging_handler_host.h" | 5 #include "chrome/browser/media/webrtc_logging_handler_host.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 | 8 |
9 #include "base/bind.h" | 9 #include "base/bind.h" |
10 #include "base/command_line.h" | 10 #include "base/command_line.h" |
11 #include "base/cpu.h" | 11 #include "base/cpu.h" |
12 #include "base/file_util.h" | 12 #include "base/file_util.h" |
13 #include "base/logging.h" | 13 #include "base/logging.h" |
14 #include "base/prefs/pref_service.h" | 14 #include "base/prefs/pref_service.h" |
15 #include "base/strings/string_number_conversions.h" | 15 #include "base/strings/string_number_conversions.h" |
16 #include "base/sys_info.h" | 16 #include "base/sys_info.h" |
17 #include "base/time/time.h" | 17 #include "base/time/time.h" |
18 #include "chrome/browser/browser_process.h" | 18 #include "chrome/browser/browser_process.h" |
19 #include "chrome/browser/chromeos/settings/cros_settings.h" | 19 #include "chrome/browser/chromeos/settings/cros_settings.h" |
20 #include "chrome/browser/media/webrtc_log_list.h" | 20 #include "chrome/browser/media/webrtc_log_list.h" |
21 #include "chrome/browser/media/webrtc_log_uploader.h" | 21 #include "chrome/browser/media/webrtc_log_uploader.h" |
22 #include "chrome/browser/media/webrtc_rtp_dump_handler.h" | |
22 #include "chrome/browser/profiles/profile.h" | 23 #include "chrome/browser/profiles/profile.h" |
23 #include "chrome/common/chrome_switches.h" | 24 #include "chrome/common/chrome_switches.h" |
24 #include "chrome/common/media/webrtc_logging_messages.h" | 25 #include "chrome/common/media/webrtc_logging_messages.h" |
25 #include "chrome/common/partial_circular_buffer.h" | 26 #include "chrome/common/partial_circular_buffer.h" |
26 #include "chrome/common/pref_names.h" | 27 #include "chrome/common/pref_names.h" |
27 #include "chromeos/settings/cros_settings_names.h" | 28 #include "chromeos/settings/cros_settings_names.h" |
28 #include "content/public/browser/browser_thread.h" | 29 #include "content/public/browser/browser_thread.h" |
29 #include "content/public/browser/content_browser_client.h" | 30 #include "content/public/browser/content_browser_client.h" |
30 #include "content/public/browser/gpu_data_manager.h" | 31 #include "content/public/browser/gpu_data_manager.h" |
31 #include "content/public/browser/render_process_host.h" | 32 #include "content/public/browser/render_process_host.h" |
32 #include "gpu/config/gpu_info.h" | 33 #include "gpu/config/gpu_info.h" |
33 #include "net/base/address_family.h" | 34 #include "net/base/address_family.h" |
34 #include "net/url_request/url_request_context_getter.h" | 35 #include "net/url_request/url_request_context_getter.h" |
35 | 36 |
36 #if defined(OS_LINUX) | 37 #if defined(OS_LINUX) |
37 #include "base/linux_util.h" | 38 #include "base/linux_util.h" |
38 #endif | 39 #endif |
39 | 40 |
40 #if defined(OS_MACOSX) | 41 #if defined(OS_MACOSX) |
41 #include "base/mac/mac_util.h" | 42 #include "base/mac/mac_util.h" |
42 #endif | 43 #endif |
43 | 44 |
44 #if defined(OS_CHROMEOS) | 45 #if defined(OS_CHROMEOS) |
45 #include "chromeos/system/statistics_provider.h" | 46 #include "chromeos/system/statistics_provider.h" |
46 #endif | 47 #endif |
47 | 48 |
48 using base::IntToString; | 49 using base::IntToString; |
49 using content::BrowserThread; | 50 using content::BrowserThread; |
50 | 51 |
52 namespace { | |
51 | 53 |
52 #if defined(OS_ANDROID) | 54 #if defined(OS_ANDROID) |
53 const size_t kWebRtcLogSize = 1 * 1024 * 1024; // 1 MB | 55 const size_t kWebRtcLogSize = 1 * 1024 * 1024; // 1 MB |
54 #else | 56 #else |
55 const size_t kWebRtcLogSize = 6 * 1024 * 1024; // 6 MB | 57 const size_t kWebRtcLogSize = 6 * 1024 * 1024; // 6 MB |
56 #endif | 58 #endif |
57 | 59 |
58 namespace { | |
59 | |
60 const char kLogNotStoppedOrNoLogOpen[] = | 60 const char kLogNotStoppedOrNoLogOpen[] = |
61 "Logging not stopped or no log open."; | 61 "Logging not stopped or no log open."; |
62 | 62 |
63 // For privacy reasons when logging IP addresses. The returned "sensitive | 63 // For privacy reasons when logging IP addresses. The returned "sensitive |
64 // string" is for release builds a string with the end stripped away. Last | 64 // string" is for release builds a string with the end stripped away. Last |
65 // octet for IPv4 and last 80 bits (5 groups) for IPv6. String will be | 65 // octet for IPv4 and last 80 bits (5 groups) for IPv6. String will be |
66 // "1.2.3.x" and "1.2.3::" respectively. For debug builds, the string is | 66 // "1.2.3.x" and "1.2.3::" respectively. For debug builds, the string is |
67 // not stripped. | 67 // not stripped. |
68 std::string IPAddressToSensitiveString(const net::IPAddressNumber& address) { | 68 std::string IPAddressToSensitiveString(const net::IPAddressNumber& address) { |
69 #if defined(NDEBUG) | 69 #if defined(NDEBUG) |
(...skipping 106 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
176 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 176 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
177 DCHECK(!callback.is_null()); | 177 DCHECK(!callback.is_null()); |
178 | 178 |
179 if (logging_state_ != STOPPED) { | 179 if (logging_state_ != STOPPED) { |
180 if (!callback.is_null()) { | 180 if (!callback.is_null()) { |
181 content::BrowserThread::PostTask(content::BrowserThread::UI, FROM_HERE, | 181 content::BrowserThread::PostTask(content::BrowserThread::UI, FROM_HERE, |
182 base::Bind(callback, false, "", kLogNotStoppedOrNoLogOpen)); | 182 base::Bind(callback, false, "", kLogNotStoppedOrNoLogOpen)); |
183 } | 183 } |
184 return; | 184 return; |
185 } | 185 } |
186 if (rtp_dump_handler_ && !rtp_dump_handler_->ReadyToRelease()) { | |
187 if (!callback.is_null()) { | |
188 content::BrowserThread::PostTask( | |
189 content::BrowserThread::UI, | |
190 FROM_HERE, | |
191 base::Bind(callback, false, "", "RTP dumping has not been stopped.")); | |
192 } | |
193 return; | |
194 } | |
195 | |
186 upload_callback_ = callback; | 196 upload_callback_ = callback; |
187 logging_state_ = UPLOADING; | 197 logging_state_ = UPLOADING; |
188 content::BrowserThread::PostTaskAndReplyWithResult( | 198 content::BrowserThread::PostTaskAndReplyWithResult( |
189 content::BrowserThread::FILE, | 199 content::BrowserThread::FILE, |
190 FROM_HERE, | 200 FROM_HERE, |
191 base::Bind(&WebRtcLoggingHandlerHost::GetLogDirectoryAndEnsureExists, | 201 base::Bind(&WebRtcLoggingHandlerHost::GetLogDirectoryAndEnsureExists, |
192 this), | 202 this), |
193 base::Bind(&WebRtcLoggingHandlerHost::TriggerUploadLog, this)); | 203 base::Bind(&WebRtcLoggingHandlerHost::TriggerUpload, this)); |
194 } | 204 } |
195 | 205 |
196 void WebRtcLoggingHandlerHost::UploadLogDone() { | 206 void WebRtcLoggingHandlerHost::UploadLogDone() { |
197 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 207 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
198 logging_state_ = CLOSED; | 208 logging_state_ = CLOSED; |
199 } | 209 } |
200 | 210 |
201 void WebRtcLoggingHandlerHost::DiscardLog(const GenericDoneCallback& callback) { | 211 void WebRtcLoggingHandlerHost::DiscardLog(const GenericDoneCallback& callback) { |
202 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 212 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
203 DCHECK(!callback.is_null()); | 213 DCHECK(!callback.is_null()); |
204 | 214 |
205 GenericDoneCallback discard_callback = callback; | 215 GenericDoneCallback discard_callback = callback; |
206 if (logging_state_ != STOPPED) { | 216 if (logging_state_ != STOPPED) { |
207 FireGenericDoneCallback(&discard_callback, false, | 217 FireGenericDoneCallback(&discard_callback, false, |
208 kLogNotStoppedOrNoLogOpen); | 218 kLogNotStoppedOrNoLogOpen); |
209 return; | 219 return; |
210 } | 220 } |
211 g_browser_process->webrtc_log_uploader()->LoggingStoppedDontUpload(); | 221 g_browser_process->webrtc_log_uploader()->LoggingStoppedDontUpload(); |
212 circular_buffer_.reset(); | 222 circular_buffer_.reset(); |
213 log_buffer_.reset(); | 223 log_buffer_.reset(); |
214 logging_state_ = CLOSED; | 224 logging_state_ = CLOSED; |
225 rtp_dump_handler_.reset(); | |
215 FireGenericDoneCallback(&discard_callback, true, ""); | 226 FireGenericDoneCallback(&discard_callback, true, ""); |
216 } | 227 } |
217 | 228 |
218 void WebRtcLoggingHandlerHost::LogMessage(const std::string& message) { | 229 void WebRtcLoggingHandlerHost::LogMessage(const std::string& message) { |
219 BrowserThread::PostTask( | 230 BrowserThread::PostTask( |
220 BrowserThread::IO, | 231 BrowserThread::IO, |
221 FROM_HERE, | 232 FROM_HERE, |
222 base::Bind( | 233 base::Bind( |
223 &WebRtcLoggingHandlerHost::AddLogMessageFromBrowser, | 234 &WebRtcLoggingHandlerHost::AddLogMessageFromBrowser, |
224 this, | 235 this, |
225 WebRtcLoggingMessageData(base::Time::Now(), message))); | 236 WebRtcLoggingMessageData(base::Time::Now(), message))); |
226 } | 237 } |
227 | 238 |
228 void WebRtcLoggingHandlerHost::StartRtpDump( | 239 void WebRtcLoggingHandlerHost::StartRtpDump( |
229 bool incoming, | 240 RtpDumpType type, |
230 bool outgoing, | 241 const GenericDoneCallback& callback, |
231 const GenericDoneCallback& callback) { | 242 const content::RenderProcessHost::WebRtcStopRtpDumpCallback& |
232 NOTIMPLEMENTED(); | 243 stop_callback) { |
244 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
245 | |
246 stop_rtp_dump_callback_ = stop_callback; | |
vrk (LEFT CHROMIUM)
2014/05/28 23:55:07
Add dcheck to make sure this is the same for incom
| |
247 | |
248 if (!rtp_dump_handler_) { | |
249 content::BrowserThread::PostTaskAndReplyWithResult( | |
250 content::BrowserThread::FILE, | |
251 FROM_HERE, | |
252 base::Bind(&WebRtcLoggingHandlerHost::GetLogDirectoryAndEnsureExists, | |
253 this), | |
254 base::Bind(&WebRtcLoggingHandlerHost::CreateRtpDumpHandlerAndStart, | |
255 this, | |
256 type, | |
257 callback)); | |
258 return; | |
259 } | |
260 | |
261 GenericDoneCallback start_callback = callback; | |
262 DoStartRtpDump(type, &start_callback); | |
233 } | 263 } |
234 | 264 |
235 void WebRtcLoggingHandlerHost::StopRtpDump( | 265 void WebRtcLoggingHandlerHost::StopRtpDump( |
236 bool incoming, | 266 RtpDumpType type, |
237 bool outgoing, | |
238 const GenericDoneCallback& callback) { | 267 const GenericDoneCallback& callback) { |
239 NOTIMPLEMENTED(); | 268 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
269 DCHECK(!callback.is_null()); | |
270 | |
271 if (!rtp_dump_handler_) { | |
272 GenericDoneCallback stop_callback = callback; | |
273 FireGenericDoneCallback( | |
274 &stop_callback, false, "RTP dump has not been started."); | |
275 return; | |
276 } | |
277 | |
278 if (!stop_rtp_dump_callback_.is_null()) { | |
279 BrowserThread::PostTask( | |
280 BrowserThread::UI, | |
281 FROM_HERE, | |
282 base::Bind(stop_rtp_dump_callback_, | |
283 type == RTP_DUMP_INCOMING || type == RTP_DUMP_BOTH, | |
284 type == RTP_DUMP_OUTGOING || type == RTP_DUMP_BOTH)); | |
285 } | |
286 | |
287 rtp_dump_handler_->StopDump(type, callback); | |
288 } | |
289 | |
290 void WebRtcLoggingHandlerHost::OnRtpPacket(const uint8* packet_header, | |
291 size_t header_length, | |
292 size_t packet_length, | |
293 bool incoming) { | |
294 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
295 | |
296 scoped_ptr<uint8[]> header_data(new uint8[header_length]); | |
297 memcpy(header_data.get(), packet_header, header_length); | |
298 | |
299 BrowserThread::PostTask( | |
300 BrowserThread::IO, | |
301 FROM_HERE, | |
302 base::Bind(&WebRtcLoggingHandlerHost::DumpRtpPacketOnIOThread, | |
303 this, | |
304 base::Passed(&header_data), | |
305 header_length, | |
306 packet_length, | |
307 incoming)); | |
308 } | |
309 | |
310 void WebRtcLoggingHandlerHost::DumpRtpPacketOnIOThread( | |
311 scoped_ptr<uint8[]> packet_header, | |
312 size_t header_length, | |
313 size_t packet_length, | |
314 bool incoming) { | |
315 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
316 | |
317 if (rtp_dump_handler_) { | |
vrk (LEFT CHROMIUM)
2014/05/28 23:55:07
Add comment explaining when it is null.
| |
318 rtp_dump_handler_->OnRtpPacket( | |
319 packet_header.get(), header_length, packet_length, incoming); | |
320 } | |
240 } | 321 } |
241 | 322 |
242 void WebRtcLoggingHandlerHost::OnChannelClosing() { | 323 void WebRtcLoggingHandlerHost::OnChannelClosing() { |
243 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 324 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
244 if (logging_state_ == STARTED || logging_state_ == STOPPED) { | 325 if (logging_state_ == STARTED || logging_state_ == STOPPED) { |
245 if (upload_log_on_render_close_) { | 326 if (upload_log_on_render_close_) { |
246 logging_state_ = UPLOADING; | 327 logging_state_ = UPLOADING; |
247 logging_started_time_ = base::Time(); | 328 logging_started_time_ = base::Time(); |
248 content::BrowserThread::PostTaskAndReplyWithResult( | 329 |
249 content::BrowserThread::FILE, | 330 if (!rtp_dump_handler_) { |
250 FROM_HERE, | 331 content::BrowserThread::PostTaskAndReplyWithResult( |
251 base::Bind(&WebRtcLoggingHandlerHost::GetLogDirectoryAndEnsureExists, | 332 content::BrowserThread::FILE, |
252 this), | 333 FROM_HERE, |
253 base::Bind(&WebRtcLoggingHandlerHost::TriggerUploadLog, this)); | 334 base::Bind( |
335 &WebRtcLoggingHandlerHost::GetLogDirectoryAndEnsureExists, | |
336 this), | |
337 base::Bind(&WebRtcLoggingHandlerHost::TriggerUpload, this)); | |
338 } else { | |
339 // The log directory has been ensured existing if |rtp_dump_handler_| is | |
vrk (LEFT CHROMIUM)
2014/05/28 23:55:07
nit: This is pretty subtle. I would just keep the
| |
340 // not NULL. | |
341 rtp_dump_handler_->StopOngoingDumps( | |
342 base::Bind(&WebRtcLoggingHandlerHost::TriggerUpload, | |
343 this, | |
344 WebRtcLogList::GetWebRtcLogDirectoryForProfile( | |
345 profile_->GetPath()))); | |
346 } | |
254 } else { | 347 } else { |
255 g_browser_process->webrtc_log_uploader()->LoggingStoppedDontUpload(); | 348 g_browser_process->webrtc_log_uploader()->LoggingStoppedDontUpload(); |
256 } | 349 } |
257 } | 350 } |
258 content::BrowserMessageFilter::OnChannelClosing(); | 351 content::BrowserMessageFilter::OnChannelClosing(); |
259 } | 352 } |
260 | 353 |
261 void WebRtcLoggingHandlerHost::OnDestruct() const { | 354 void WebRtcLoggingHandlerHost::OnDestruct() const { |
262 BrowserThread::DeleteOnIOThread::Destruct(this); | 355 BrowserThread::DeleteOnIOThread::Destruct(this); |
263 } | 356 } |
(...skipping 181 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
445 base::FilePath log_dir_path = | 538 base::FilePath log_dir_path = |
446 WebRtcLogList::GetWebRtcLogDirectoryForProfile(profile_->GetPath()); | 539 WebRtcLogList::GetWebRtcLogDirectoryForProfile(profile_->GetPath()); |
447 base::File::Error error; | 540 base::File::Error error; |
448 if (!base::CreateDirectoryAndGetError(log_dir_path, &error)) { | 541 if (!base::CreateDirectoryAndGetError(log_dir_path, &error)) { |
449 DLOG(ERROR) << "Could not create WebRTC log directory, error: " << error; | 542 DLOG(ERROR) << "Could not create WebRTC log directory, error: " << error; |
450 return base::FilePath(); | 543 return base::FilePath(); |
451 } | 544 } |
452 return log_dir_path; | 545 return log_dir_path; |
453 } | 546 } |
454 | 547 |
455 void WebRtcLoggingHandlerHost::TriggerUploadLog( | 548 void WebRtcLoggingHandlerHost::TriggerUpload( |
456 const base::FilePath& log_directory) { | 549 const base::FilePath& log_directory) { |
457 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 550 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
458 DCHECK_EQ(logging_state_, UPLOADING); | 551 DCHECK_EQ(logging_state_, UPLOADING); |
459 | 552 |
553 DoUploadLogAndDumps(log_directory, | |
vrk (LEFT CHROMIUM)
2014/05/28 23:55:07
Inline this method
| |
554 rtp_dump_handler_ | |
555 ? rtp_dump_handler_->ReleaseDumps() | |
556 : WebRtcRtpDumpHandler::ReleasedDumps( | |
557 base::FilePath(), base::FilePath())); | |
558 } | |
559 | |
560 void WebRtcLoggingHandlerHost::FireGenericDoneCallback( | |
561 GenericDoneCallback* callback, | |
562 bool success, | |
563 const std::string& error_message) { | |
564 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
565 DCHECK(!(*callback).is_null()); | |
566 content::BrowserThread::PostTask( | |
567 content::BrowserThread::UI, | |
568 FROM_HERE, | |
569 base::Bind(*callback, success, error_message)); | |
570 (*callback).Reset(); | |
571 } | |
572 | |
573 void WebRtcLoggingHandlerHost::CreateRtpDumpHandlerAndStart( | |
574 RtpDumpType type, | |
575 GenericDoneCallback callback, | |
576 const base::FilePath& dump_dir) { | |
577 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
578 | |
579 // |rtp_dump_handler_| may be non-NULL if StartRtpDump is called again before | |
580 // GetLogDirectoryAndEnsureExists returns on the FILE thread for a previous | |
581 // StartRtpDump. | |
582 if (!rtp_dump_handler_) | |
583 rtp_dump_handler_.reset(new WebRtcRtpDumpHandler(dump_dir)); | |
584 | |
585 DoStartRtpDump(type, &callback); | |
586 } | |
587 | |
588 void WebRtcLoggingHandlerHost::DoStartRtpDump(RtpDumpType type, | |
589 GenericDoneCallback* callback) { | |
590 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
591 DCHECK(rtp_dump_handler_); | |
592 | |
593 std::string error; | |
594 | |
595 bool result = rtp_dump_handler_->StartDump(type, &error); | |
596 FireGenericDoneCallback(callback, result, error); | |
597 } | |
598 | |
599 void WebRtcLoggingHandlerHost::DoUploadLogAndDumps( | |
600 const base::FilePath& log_directory, | |
601 const WebRtcRtpDumpHandler::ReleasedDumps& rtp_dumps) { | |
460 WebRtcLogUploadDoneData upload_done_data; | 602 WebRtcLogUploadDoneData upload_done_data; |
461 upload_done_data.log_path = log_directory; | 603 upload_done_data.log_path = log_directory; |
604 | |
605 if (!rtp_dumps.incoming_dump_path.empty()) { | |
606 upload_done_data.rtp_dumps.push_back( | |
607 WebRtcRtpDumpDescription("rtpdump_recv", rtp_dumps.incoming_dump_path)); | |
vrk (LEFT CHROMIUM)
2014/05/28 23:55:07
I think you should define these names in webrtc_lo
| |
608 } | |
609 | |
610 if (!rtp_dumps.outgoing_dump_path.empty()) { | |
611 upload_done_data.rtp_dumps.push_back( | |
612 WebRtcRtpDumpDescription("rtpdump_send", rtp_dumps.outgoing_dump_path)); | |
613 } | |
614 | |
462 upload_done_data.callback = upload_callback_; | 615 upload_done_data.callback = upload_callback_; |
463 upload_done_data.host = this; | 616 upload_done_data.host = this; |
464 upload_callback_.Reset(); | 617 upload_callback_.Reset(); |
465 | 618 |
466 BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE, base::Bind( | 619 BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE, base::Bind( |
467 &WebRtcLogUploader::LoggingStoppedDoUpload, | 620 &WebRtcLogUploader::LoggingStoppedDoUpload, |
468 base::Unretained(g_browser_process->webrtc_log_uploader()), | 621 base::Unretained(g_browser_process->webrtc_log_uploader()), |
469 Passed(&log_buffer_), | 622 Passed(&log_buffer_), |
470 kWebRtcLogSize, | 623 kWebRtcLogSize, |
471 meta_data_, | 624 meta_data_, |
472 upload_done_data)); | 625 upload_done_data)); |
473 | 626 |
474 meta_data_.clear(); | 627 meta_data_.clear(); |
475 circular_buffer_.reset(); | 628 circular_buffer_.reset(); |
476 } | 629 } |
477 | |
478 void WebRtcLoggingHandlerHost::FireGenericDoneCallback( | |
479 GenericDoneCallback* callback, bool success, | |
480 const std::string& error_message) { | |
481 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
482 DCHECK(!(*callback).is_null()); | |
483 content::BrowserThread::PostTask(content::BrowserThread::UI, FROM_HERE, | |
484 base::Bind(*callback, success, | |
485 error_message)); | |
486 (*callback).Reset(); | |
487 } | |
OLD | NEW |