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

Side by Side Diff: media/blink/webmediaplayer_impl.cc

Issue 2568303002: Report WMPI components memory usage to UMA (Closed)
Patch Set: Call ReportMemoryUsage from WMPI::OnMemoryPressure Created 4 years 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
OLDNEW
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 "media/blink/webmediaplayer_impl.h" 5 #include "media/blink/webmediaplayer_impl.h"
6 6
7 #include <algorithm> 7 #include <algorithm>
8 #include <cmath> 8 #include <cmath>
9 #include <limits> 9 #include <limits>
10 #include <string> 10 #include <string>
11 #include <utility> 11 #include <utility>
12 12
13 #include "base/bind.h" 13 #include "base/bind.h"
14 #include "base/bind_helpers.h" 14 #include "base/bind_helpers.h"
15 #include "base/callback.h" 15 #include "base/callback.h"
16 #include "base/callback_helpers.h" 16 #include "base/callback_helpers.h"
17 #include "base/command_line.h" 17 #include "base/command_line.h"
18 #include "base/debug/alias.h" 18 #include "base/debug/alias.h"
19 #include "base/debug/crash_logging.h" 19 #include "base/debug/crash_logging.h"
20 #include "base/location.h" 20 #include "base/location.h"
21 #include "base/memory/ptr_util.h"
21 #include "base/metrics/histogram_macros.h" 22 #include "base/metrics/histogram_macros.h"
22 #include "base/single_thread_task_runner.h" 23 #include "base/single_thread_task_runner.h"
23 #include "base/strings/string_number_conversions.h" 24 #include "base/strings/string_number_conversions.h"
24 #include "base/task_runner_util.h" 25 #include "base/task_runner_util.h"
25 #include "base/threading/thread_task_runner_handle.h" 26 #include "base/threading/thread_task_runner_handle.h"
26 #include "base/trace_event/trace_event.h" 27 #include "base/trace_event/trace_event.h"
27 #include "build/build_config.h" 28 #include "build/build_config.h"
28 #include "cc/blink/web_layer_impl.h" 29 #include "cc/blink/web_layer_impl.h"
29 #include "cc/layers/video_layer.h" 30 #include "cc/layers/video_layer.h"
30 #include "media/audio/null_audio_sink.h" 31 #include "media/audio/null_audio_sink.h"
(...skipping 1042 matching lines...) Expand 10 before | Expand all | Expand 10 after
1073 #endif 1074 #endif
1074 1075
1075 // If we're not in an aggressive buffering state, tell the data source we have 1076 // If we're not in an aggressive buffering state, tell the data source we have
1076 // enough data so that it may release the connection. 1077 // enough data so that it may release the connection.
1077 if (buffering_strategy_ != 1078 if (buffering_strategy_ !=
1078 MultibufferDataSource::BUFFERING_STRATEGY_AGGRESSIVE) { 1079 MultibufferDataSource::BUFFERING_STRATEGY_AGGRESSIVE) {
1079 if (data_source_) 1080 if (data_source_)
1080 data_source_->OnBufferingHaveEnough(true); 1081 data_source_->OnBufferingHaveEnough(true);
1081 } 1082 }
1082 1083
1083 ReportMemoryUsage(); 1084 ReportMemoryUsage(MaybeMemPressureLvl());
1084 1085
1085 if (pending_suspend_resume_cycle_) { 1086 if (pending_suspend_resume_cycle_) {
1086 pending_suspend_resume_cycle_ = false; 1087 pending_suspend_resume_cycle_ = false;
1087 UpdatePlayState(); 1088 UpdatePlayState();
1088 } 1089 }
1089 } 1090 }
1090 1091
1091 void WebMediaPlayerImpl::OnDemuxerOpened() { 1092 void WebMediaPlayerImpl::OnDemuxerOpened() {
1092 DCHECK(main_task_runner_->BelongsToCurrentThread()); 1093 DCHECK(main_task_runner_->BelongsToCurrentThread());
1093 client_->mediaSourceOpened( 1094 client_->mediaSourceOpened(
1094 new WebMediaSourceImpl(chunk_demuxer_, media_log_)); 1095 new WebMediaSourceImpl(chunk_demuxer_, media_log_));
1096 // Using base::Unretained(this) is safe here, because MemoryPressureListener
1097 // object is owned by this (WMPI::mem_pressure_listener_).
watk 2016/12/13 23:34:31 "because the". Or if you want to reword it: "base
servolk 2016/12/14 00:07:23 Done.
1098 mem_pressure_listener_ =
1099 base::WrapUnique(new base::MemoryPressureListener(base::Bind(
watk 2016/12/13 23:34:31 Prefer MakeUnique(). (Unless there's a reason you
servolk 2016/12/14 00:07:23 Done.
1100 &WebMediaPlayerImpl::OnMemoryPressure, base::Unretained(this))));
1101 }
1102
1103 void WebMediaPlayerImpl::OnMemoryPressure(
1104 base::MemoryPressureListener::MemoryPressureLevel lvl) {
1105 DCHECK(main_task_runner_->BelongsToCurrentThread());
1106 ReportMemoryUsage(lvl);
1095 } 1107 }
1096 1108
1097 void WebMediaPlayerImpl::OnError(PipelineStatus status) { 1109 void WebMediaPlayerImpl::OnError(PipelineStatus status) {
1098 DVLOG(1) << __func__; 1110 DVLOG(1) << __func__;
1099 DCHECK(main_task_runner_->BelongsToCurrentThread()); 1111 DCHECK(main_task_runner_->BelongsToCurrentThread());
1100 DCHECK_NE(status, PIPELINE_OK); 1112 DCHECK_NE(status, PIPELINE_OK);
1101 1113
1102 if (suppress_destruction_errors_) 1114 if (suppress_destruction_errors_)
1103 return; 1115 return;
1104 1116
(...skipping 87 matching lines...) Expand 10 before | Expand all | Expand 10 after
1192 // to release unused network connections. 1204 // to release unused network connections.
1193 if (data_source_) 1205 if (data_source_)
1194 data_source_->OnBufferingHaveEnough(false); 1206 data_source_->OnBufferingHaveEnough(false);
1195 1207
1196 // Blink expects a timeChanged() in response to a seek(). 1208 // Blink expects a timeChanged() in response to a seek().
1197 if (should_notify_time_changed_) 1209 if (should_notify_time_changed_)
1198 client_->timeChanged(); 1210 client_->timeChanged();
1199 1211
1200 // Once we have enough, start reporting the total memory usage. We'll also 1212 // Once we have enough, start reporting the total memory usage. We'll also
1201 // report once playback starts. 1213 // report once playback starts.
1202 ReportMemoryUsage(); 1214 ReportMemoryUsage(MaybeMemPressureLvl());
1203 1215
1204 // Report the amount of time it took to leave the underflow state. Don't 1216 // Report the amount of time it took to leave the underflow state. Don't
1205 // bother to report this for MSE playbacks since it's out of our control. 1217 // bother to report this for MSE playbacks since it's out of our control.
1206 if (underflow_timer_ && data_source_) { 1218 if (underflow_timer_ && data_source_) {
1207 UMA_HISTOGRAM_TIMES("Media.UnderflowDuration", 1219 UMA_HISTOGRAM_TIMES("Media.UnderflowDuration",
1208 underflow_timer_->Elapsed()); 1220 underflow_timer_->Elapsed());
1209 underflow_timer_.reset(); 1221 underflow_timer_.reset();
1210 } 1222 }
1211 } else { 1223 } else {
1212 // Buffering has underflowed. 1224 // Buffering has underflowed.
(...skipping 544 matching lines...) Expand 10 before | Expand all | Expand 10 after
1757 } 1769 }
1758 1770
1759 void WebMediaPlayerImpl::SetMemoryReportingState( 1771 void WebMediaPlayerImpl::SetMemoryReportingState(
1760 bool is_memory_reporting_enabled) { 1772 bool is_memory_reporting_enabled) {
1761 if (memory_usage_reporting_timer_.IsRunning() == 1773 if (memory_usage_reporting_timer_.IsRunning() ==
1762 is_memory_reporting_enabled) { 1774 is_memory_reporting_enabled) {
1763 return; 1775 return;
1764 } 1776 }
1765 1777
1766 if (is_memory_reporting_enabled) { 1778 if (is_memory_reporting_enabled) {
1767 memory_usage_reporting_timer_.Start(FROM_HERE, 1779 memory_usage_reporting_timer_.Start(
1768 base::TimeDelta::FromSeconds(2), this, 1780 FROM_HERE, base::TimeDelta::FromSeconds(2), this,
1769 &WebMediaPlayerImpl::ReportMemoryUsage); 1781 &WebMediaPlayerImpl::ReportMemoryUsageOnTimer);
1770 } else { 1782 } else {
1771 memory_usage_reporting_timer_.Stop(); 1783 memory_usage_reporting_timer_.Stop();
1772 ReportMemoryUsage(); 1784 ReportMemoryUsage(MaybeMemPressureLvl());
1773 } 1785 }
1774 } 1786 }
1775 1787
1776 void WebMediaPlayerImpl::SetSuspendState(bool is_suspended) { 1788 void WebMediaPlayerImpl::SetSuspendState(bool is_suspended) {
1777 DCHECK(main_task_runner_->BelongsToCurrentThread()); 1789 DCHECK(main_task_runner_->BelongsToCurrentThread());
1778 1790
1779 // Do not change the state after an error has occurred. 1791 // Do not change the state after an error has occurred.
1780 // TODO(sandersd): Update PipelineController to remove the need for this. 1792 // TODO(sandersd): Update PipelineController to remove the need for this.
1781 if (IsNetworkStateError(network_state_)) 1793 if (IsNetworkStateError(network_state_))
1782 return; 1794 return;
(...skipping 120 matching lines...) Expand 10 before | Expand all | Expand 10 after
1903 } 1915 }
1904 1916
1905 // It's not critical if some cases where memory usage can change are missed, 1917 // It's not critical if some cases where memory usage can change are missed,
1906 // since media memory changes are usually gradual. 1918 // since media memory changes are usually gradual.
1907 result.is_memory_reporting_enabled = 1919 result.is_memory_reporting_enabled =
1908 can_play && !result.is_suspended && !paused_; 1920 can_play && !result.is_suspended && !paused_;
1909 1921
1910 return result; 1922 return result;
1911 } 1923 }
1912 1924
1913 void WebMediaPlayerImpl::ReportMemoryUsage() { 1925 void WebMediaPlayerImpl::ReportMemoryUsageOnTimer() {
watk 2016/12/13 23:34:31 Can you bind the nullopt into the callback and del
servolk 2016/12/14 00:07:23 This is passed into timer.Start (see https://cs.ch
servolk 2016/12/14 00:11:23 Ah, wait, despite the weird Start method at https:
servolk 2016/12/14 00:15:37 Done.
1926 ReportMemoryUsage(MaybeMemPressureLvl());
1927 }
1928
1929 void WebMediaPlayerImpl::ReportMemoryUsage(
1930 MaybeMemPressureLvl maybe_mem_pressure_lvl) {
1914 DCHECK(main_task_runner_->BelongsToCurrentThread()); 1931 DCHECK(main_task_runner_->BelongsToCurrentThread());
1915 1932
1916 // About base::Unretained() usage below: We destroy |demuxer_| on the main 1933 // About base::Unretained() usage below: We destroy |demuxer_| on the main
1917 // thread. Before that, however, ~WebMediaPlayerImpl() posts a task to the 1934 // thread. Before that, however, ~WebMediaPlayerImpl() posts a task to the
1918 // media thread and waits for it to finish. Hence, the GetMemoryUsage() task 1935 // media thread and waits for it to finish. Hence, the GetMemoryUsage() task
1919 // posted here must finish earlier. 1936 // posted here must finish earlier.
1920 1937
1921 if (demuxer_) { 1938 if (demuxer_) {
1922 base::PostTaskAndReplyWithResult( 1939 base::PostTaskAndReplyWithResult(
1923 media_task_runner_.get(), FROM_HERE, 1940 media_task_runner_.get(), FROM_HERE,
1924 base::Bind(&Demuxer::GetMemoryUsage, base::Unretained(demuxer_.get())), 1941 base::Bind(&Demuxer::GetMemoryUsage, base::Unretained(demuxer_.get())),
1925 base::Bind(&WebMediaPlayerImpl::FinishMemoryUsageReport, AsWeakPtr())); 1942 base::Bind(&WebMediaPlayerImpl::FinishMemoryUsageReport, AsWeakPtr(),
1943 maybe_mem_pressure_lvl));
1926 } else { 1944 } else {
1927 FinishMemoryUsageReport(0); 1945 FinishMemoryUsageReport(maybe_mem_pressure_lvl, 0);
1928 } 1946 }
1929 } 1947 }
1930 1948
1931 void WebMediaPlayerImpl::FinishMemoryUsageReport(int64_t demuxer_memory_usage) { 1949 std::string MaybeMemPressureToLogString(
1950 const base::Optional<base::MemoryPressureListener::MemoryPressureLevel>&
watk 2016/12/13 23:34:31 Use the type alias?
servolk 2016/12/14 00:07:23 I could, but the type alias is currently private i
watk 2016/12/14 00:34:05 Ah, didn't notice this was a non-member function.
servolk 2016/12/14 02:34:33 Acknowledged.
1951 maybe_mem_pressure_lvl) {
1952 if (maybe_mem_pressure_lvl) {
1953 switch (*maybe_mem_pressure_lvl) {
1954 case base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_NONE:
1955 return "none";
1956 case base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE:
1957 return "moderate";
1958 case base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL:
1959 return "critical";
1960 }
1961 }
1962 return "";
1963 }
1964
1965 void WebMediaPlayerImpl::FinishMemoryUsageReport(
1966 MaybeMemPressureLvl maybe_mem_pressure_lvl,
1967 int64_t demuxer_memory_usage) {
1932 DCHECK(main_task_runner_->BelongsToCurrentThread()); 1968 DCHECK(main_task_runner_->BelongsToCurrentThread());
1933 1969
1934 const PipelineStatistics stats = pipeline_.GetStatistics(); 1970 const PipelineStatistics stats = pipeline_.GetStatistics();
1971 const int64_t data_source_mem_usage =
watk 2016/12/13 23:34:31 "memory_usage" for consistency with other variable
servolk 2016/12/14 00:07:23 Done.
1972 (data_source_ ? data_source_->GetMemoryUsage() : 0);
1935 const int64_t current_memory_usage = 1973 const int64_t current_memory_usage =
1936 stats.audio_memory_usage + stats.video_memory_usage + 1974 stats.audio_memory_usage + stats.video_memory_usage +
1937 (data_source_ ? data_source_->GetMemoryUsage() : 0) + 1975 data_source_mem_usage + demuxer_memory_usage;
1938 demuxer_memory_usage; 1976
1977 std::string mem_pressure_log =
1978 MaybeMemPressureToLogString(maybe_mem_pressure_lvl);
1979 if (mem_pressure_log != "")
1980 mem_pressure_log = " (Mem pressure: " + mem_pressure_log + ")";
watk 2016/12/13 23:34:31 Do this inside MaybeMemPressureToLogString and inl
servolk 2016/12/14 00:07:23 This would make MaybeMemPressureToLogString less r
1939 1981
1940 // Note, this isn't entirely accurate, there may be VideoFrames held by the 1982 // Note, this isn't entirely accurate, there may be VideoFrames held by the
1941 // compositor or other resources that we're unaware of. 1983 // compositor or other resources that we're unaware of.
1942 1984 DVLOG(2) << "Memory Usage " << mem_pressure_log
1943 DVLOG(2) << "Memory Usage -- Audio: " << stats.audio_memory_usage 1985 << " -- Audio: " << stats.audio_memory_usage
1944 << ", Video: " << stats.video_memory_usage << ", DataSource: " 1986 << ", Video: " << stats.video_memory_usage
1945 << (data_source_ ? data_source_->GetMemoryUsage() : 0) 1987 << ", DataSource: " << data_source_mem_usage
1946 << ", Demuxer: " << demuxer_memory_usage; 1988 << ", Demuxer: " << demuxer_memory_usage;
1947 1989
1948 const int64_t delta = current_memory_usage - last_reported_memory_usage_; 1990 const int64_t delta = current_memory_usage - last_reported_memory_usage_;
1949 last_reported_memory_usage_ = current_memory_usage; 1991 last_reported_memory_usage_ = current_memory_usage;
1950 adjust_allocated_memory_cb_.Run(delta); 1992 adjust_allocated_memory_cb_.Run(delta);
1993
1994 if (maybe_mem_pressure_lvl &&
DaleCurtis 2016/12/12 23:49:48 Hmm, why? Why not just always report the average m
servolk 2016/12/12 23:56:12 ReportMemoryUsage and FinishMemoryUsageReport are
DaleCurtis 2016/12/13 00:04:07 Reporting UMA is mostly just updating a static val
1995 *maybe_mem_pressure_lvl >=
1996 base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE) {
1997 UMA_HISTOGRAM_MEMORY_KB("Media.WebMediaPlayerImpl.Memory.AudioKb",
1998 stats.audio_memory_usage / 1024);
1999 UMA_HISTOGRAM_MEMORY_KB("Media.WebMediaPlayerImpl.Memory.VideoKb",
2000 stats.video_memory_usage / 1024);
2001 UMA_HISTOGRAM_MEMORY_KB("Media.WebMediaPlayerImpl.Memory.DataSourceKb",
2002 stats.video_memory_usage / 1024);
2003 UMA_HISTOGRAM_MEMORY_KB("Media.WebMediaPlayerImpl.Memory.DemuxerKb",
2004 demuxer_memory_usage / 1024);
2005 }
1951 } 2006 }
1952 2007
1953 void WebMediaPlayerImpl::ScheduleIdlePauseTimer() { 2008 void WebMediaPlayerImpl::ScheduleIdlePauseTimer() {
1954 // Only schedule the pause timer if we're playing and are suspended. 2009 // Only schedule the pause timer if we're playing and are suspended.
1955 if (paused_ || !pipeline_controller_.IsSuspended()) 2010 if (paused_ || !pipeline_controller_.IsSuspended())
1956 return; 2011 return;
1957 2012
1958 #if defined(OS_ANDROID) 2013 #if defined(OS_ANDROID)
1959 // Remote players will be suspended and locally paused. 2014 // Remote players will be suspended and locally paused.
1960 if (isRemote()) 2015 if (isRemote())
(...skipping 18 matching lines...) Expand all
1979 watch_time_reporter_->OnShown(); 2034 watch_time_reporter_->OnShown();
1980 } 2035 }
1981 2036
1982 bool WebMediaPlayerImpl::IsHidden() const { 2037 bool WebMediaPlayerImpl::IsHidden() const {
1983 DCHECK(main_task_runner_->BelongsToCurrentThread()); 2038 DCHECK(main_task_runner_->BelongsToCurrentThread());
1984 2039
1985 return delegate_ && delegate_->IsHidden(); 2040 return delegate_ && delegate_->IsHidden();
1986 } 2041 }
1987 2042
1988 } // namespace media 2043 } // namespace media
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698