OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
11 #include "webrtc/tools/event_log_visualizer/analyzer.h" | 11 #include "webrtc/tools/event_log_visualizer/analyzer.h" |
12 | 12 |
13 #include <algorithm> | 13 #include <algorithm> |
14 #include <limits> | 14 #include <limits> |
15 #include <map> | 15 #include <map> |
16 #include <sstream> | 16 #include <sstream> |
17 #include <string> | 17 #include <string> |
18 #include <utility> | 18 #include <utility> |
19 | 19 |
20 #include "webrtc/audio_receive_stream.h" | 20 #include "webrtc/audio_receive_stream.h" |
21 #include "webrtc/audio_send_stream.h" | 21 #include "webrtc/audio_send_stream.h" |
22 #include "webrtc/base/checks.h" | 22 #include "webrtc/base/checks.h" |
23 #include "webrtc/call.h" | 23 #include "webrtc/call.h" |
24 #include "webrtc/common_types.h" | 24 #include "webrtc/common_types.h" |
25 #include "webrtc/modules/congestion_controller/include/congestion_controller.h" | |
25 #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp.h" | 26 #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp.h" |
26 #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp_defines.h" | 27 #include "webrtc/modules/rtp_rtcp/include/rtp_rtcp_defines.h" |
27 #include "webrtc/modules/rtp_rtcp/source/rtp_utility.h" | 28 #include "webrtc/modules/rtp_rtcp/source/rtp_utility.h" |
29 #include "webrtc/modules/rtp_rtcp/source/rtcp_utility.h" | |
30 #include "webrtc/modules/rtp_rtcp/source/rtcp_packet/transport_feedback.h" | |
28 #include "webrtc/video_receive_stream.h" | 31 #include "webrtc/video_receive_stream.h" |
29 #include "webrtc/video_send_stream.h" | 32 #include "webrtc/video_send_stream.h" |
30 | 33 |
31 namespace { | 34 namespace { |
32 | 35 |
33 std::string SsrcToString(uint32_t ssrc) { | 36 std::string SsrcToString(uint32_t ssrc) { |
34 std::stringstream ss; | 37 std::stringstream ss; |
35 ss << "SSRC " << ssrc; | 38 ss << "SSRC " << ssrc; |
36 return ss.str(); | 39 return ss.str(); |
37 } | 40 } |
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
85 | 88 |
86 namespace webrtc { | 89 namespace webrtc { |
87 namespace plotting { | 90 namespace plotting { |
88 | 91 |
89 | 92 |
90 bool EventLogAnalyzer::StreamId::operator<(const StreamId& other) const { | 93 bool EventLogAnalyzer::StreamId::operator<(const StreamId& other) const { |
91 if (ssrc_ < other.ssrc_) { | 94 if (ssrc_ < other.ssrc_) { |
92 return true; | 95 return true; |
93 } | 96 } |
94 if (ssrc_ == other.ssrc_) { | 97 if (ssrc_ == other.ssrc_) { |
95 if (media_type_ < other.media_type_) { | 98 if (direction_ < other.direction_) { |
96 return true; | 99 return true; |
97 } | 100 } |
98 if (media_type_ == other.media_type_) { | |
99 if (direction_ < other.direction_) { | |
100 return true; | |
101 } | |
102 } | |
103 } | 101 } |
104 return false; | 102 return false; |
105 } | 103 } |
106 | 104 |
107 bool EventLogAnalyzer::StreamId::operator==(const StreamId& other) const { | 105 bool EventLogAnalyzer::StreamId::operator==(const StreamId& other) const { |
108 return ssrc_ == other.ssrc_ && direction_ == other.direction_ && | 106 return ssrc_ == other.ssrc_ && direction_ == other.direction_; |
109 media_type_ == other.media_type_; | |
110 } | 107 } |
111 | 108 |
112 | 109 |
113 EventLogAnalyzer::EventLogAnalyzer(const ParsedRtcEventLog& log) | 110 EventLogAnalyzer::EventLogAnalyzer(const ParsedRtcEventLog& log) |
114 : parsed_log_(log), window_duration_(250000), step_(10000) { | 111 : parsed_log_(log), window_duration_(250000), step_(10000) { |
115 uint64_t first_timestamp = std::numeric_limits<uint64_t>::max(); | 112 uint64_t first_timestamp = std::numeric_limits<uint64_t>::max(); |
116 uint64_t last_timestamp = std::numeric_limits<uint64_t>::min(); | 113 uint64_t last_timestamp = std::numeric_limits<uint64_t>::min(); |
117 | 114 |
118 // Maps a stream identifier consisting of ssrc, direction and MediaType | 115 // Maps a stream identifier consisting of ssrc and direction |
119 // to the header extensions used by that stream, | 116 // to the header extensions used by that stream, |
120 std::map<StreamId, RtpHeaderExtensionMap> extension_maps; | 117 std::map<StreamId, RtpHeaderExtensionMap> extension_maps; |
121 | 118 |
122 PacketDirection direction; | 119 PacketDirection direction; |
123 MediaType media_type; | |
124 uint8_t header[IP_PACKET_SIZE]; | 120 uint8_t header[IP_PACKET_SIZE]; |
125 size_t header_length; | 121 size_t header_length; |
126 size_t total_length; | 122 size_t total_length; |
127 | 123 |
128 for (size_t i = 0; i < parsed_log_.GetNumberOfEvents(); i++) { | 124 for (size_t i = 0; i < parsed_log_.GetNumberOfEvents(); i++) { |
129 ParsedRtcEventLog::EventType event_type = parsed_log_.GetEventType(i); | 125 ParsedRtcEventLog::EventType event_type = parsed_log_.GetEventType(i); |
130 if (event_type != ParsedRtcEventLog::VIDEO_RECEIVER_CONFIG_EVENT && | 126 if (event_type != ParsedRtcEventLog::VIDEO_RECEIVER_CONFIG_EVENT && |
131 event_type != ParsedRtcEventLog::VIDEO_SENDER_CONFIG_EVENT && | 127 event_type != ParsedRtcEventLog::VIDEO_SENDER_CONFIG_EVENT && |
132 event_type != ParsedRtcEventLog::AUDIO_RECEIVER_CONFIG_EVENT && | 128 event_type != ParsedRtcEventLog::AUDIO_RECEIVER_CONFIG_EVENT && |
133 event_type != ParsedRtcEventLog::AUDIO_SENDER_CONFIG_EVENT) { | 129 event_type != ParsedRtcEventLog::AUDIO_SENDER_CONFIG_EVENT) { |
134 uint64_t timestamp = parsed_log_.GetTimestamp(i); | 130 uint64_t timestamp = parsed_log_.GetTimestamp(i); |
135 first_timestamp = std::min(first_timestamp, timestamp); | 131 first_timestamp = std::min(first_timestamp, timestamp); |
136 last_timestamp = std::max(last_timestamp, timestamp); | 132 last_timestamp = std::max(last_timestamp, timestamp); |
137 } | 133 } |
138 | 134 |
139 switch (parsed_log_.GetEventType(i)) { | 135 switch (parsed_log_.GetEventType(i)) { |
140 case ParsedRtcEventLog::VIDEO_RECEIVER_CONFIG_EVENT: { | 136 case ParsedRtcEventLog::VIDEO_RECEIVER_CONFIG_EVENT: { |
141 VideoReceiveStream::Config config(nullptr); | 137 VideoReceiveStream::Config config(nullptr); |
142 parsed_log_.GetVideoReceiveConfig(i, &config); | 138 parsed_log_.GetVideoReceiveConfig(i, &config); |
143 StreamId stream(config.rtp.remote_ssrc, kIncomingPacket, | 139 StreamId stream(config.rtp.remote_ssrc, kIncomingPacket); |
144 MediaType::VIDEO); | |
145 extension_maps[stream].Erase(); | 140 extension_maps[stream].Erase(); |
146 for (size_t j = 0; j < config.rtp.extensions.size(); ++j) { | 141 for (size_t j = 0; j < config.rtp.extensions.size(); ++j) { |
147 const std::string& extension = config.rtp.extensions[j].uri; | 142 const std::string& extension = config.rtp.extensions[j].uri; |
148 int id = config.rtp.extensions[j].id; | 143 int id = config.rtp.extensions[j].id; |
149 extension_maps[stream].Register(StringToRtpExtensionType(extension), | 144 extension_maps[stream].Register(StringToRtpExtensionType(extension), |
150 id); | 145 id); |
151 } | 146 } |
152 break; | 147 break; |
153 } | 148 } |
154 case ParsedRtcEventLog::VIDEO_SENDER_CONFIG_EVENT: { | 149 case ParsedRtcEventLog::VIDEO_SENDER_CONFIG_EVENT: { |
155 VideoSendStream::Config config(nullptr); | 150 VideoSendStream::Config config(nullptr); |
156 parsed_log_.GetVideoSendConfig(i, &config); | 151 parsed_log_.GetVideoSendConfig(i, &config); |
157 for (auto ssrc : config.rtp.ssrcs) { | 152 for (auto ssrc : config.rtp.ssrcs) { |
158 StreamId stream(ssrc, kOutgoingPacket, MediaType::VIDEO); | 153 StreamId stream(ssrc, kOutgoingPacket); |
159 extension_maps[stream].Erase(); | 154 extension_maps[stream].Erase(); |
160 for (size_t j = 0; j < config.rtp.extensions.size(); ++j) { | 155 for (size_t j = 0; j < config.rtp.extensions.size(); ++j) { |
161 const std::string& extension = config.rtp.extensions[j].uri; | 156 const std::string& extension = config.rtp.extensions[j].uri; |
162 int id = config.rtp.extensions[j].id; | 157 int id = config.rtp.extensions[j].id; |
163 extension_maps[stream].Register(StringToRtpExtensionType(extension), | 158 extension_maps[stream].Register(StringToRtpExtensionType(extension), |
164 id); | 159 id); |
165 } | 160 } |
166 } | 161 } |
167 break; | 162 break; |
168 } | 163 } |
169 case ParsedRtcEventLog::AUDIO_RECEIVER_CONFIG_EVENT: { | 164 case ParsedRtcEventLog::AUDIO_RECEIVER_CONFIG_EVENT: { |
170 AudioReceiveStream::Config config; | 165 AudioReceiveStream::Config config; |
171 // TODO(terelius): Parse the audio configs once we have them. | 166 // TODO(terelius): Parse the audio configs once we have them. |
172 break; | 167 break; |
173 } | 168 } |
174 case ParsedRtcEventLog::AUDIO_SENDER_CONFIG_EVENT: { | 169 case ParsedRtcEventLog::AUDIO_SENDER_CONFIG_EVENT: { |
175 AudioSendStream::Config config(nullptr); | 170 AudioSendStream::Config config(nullptr); |
176 // TODO(terelius): Parse the audio configs once we have them. | 171 // TODO(terelius): Parse the audio configs once we have them. |
177 break; | 172 break; |
178 } | 173 } |
179 case ParsedRtcEventLog::RTP_EVENT: { | 174 case ParsedRtcEventLog::RTP_EVENT: { |
175 MediaType media_type; | |
180 parsed_log_.GetRtpHeader(i, &direction, &media_type, header, | 176 parsed_log_.GetRtpHeader(i, &direction, &media_type, header, |
181 &header_length, &total_length); | 177 &header_length, &total_length); |
182 // Parse header to get SSRC. | 178 // Parse header to get SSRC. |
183 RtpUtility::RtpHeaderParser rtp_parser(header, header_length); | 179 RtpUtility::RtpHeaderParser rtp_parser(header, header_length); |
184 RTPHeader parsed_header; | 180 RTPHeader parsed_header; |
185 rtp_parser.Parse(&parsed_header); | 181 rtp_parser.Parse(&parsed_header); |
186 StreamId stream(parsed_header.ssrc, direction, media_type); | 182 StreamId stream(parsed_header.ssrc, direction); |
187 // Look up the extension_map and parse it again to get the extensions. | 183 // Look up the extension_map and parse it again to get the extensions. |
188 if (extension_maps.count(stream) == 1) { | 184 if (extension_maps.count(stream) == 1) { |
189 RtpHeaderExtensionMap* extension_map = &extension_maps[stream]; | 185 RtpHeaderExtensionMap* extension_map = &extension_maps[stream]; |
190 rtp_parser.Parse(&parsed_header, extension_map); | 186 rtp_parser.Parse(&parsed_header, extension_map); |
191 } | 187 } |
192 uint64_t timestamp = parsed_log_.GetTimestamp(i); | 188 uint64_t timestamp = parsed_log_.GetTimestamp(i); |
193 rtp_packets_[stream].push_back( | 189 rtp_packets_[stream].push_back( |
194 LoggedRtpPacket(timestamp, parsed_header)); | 190 LoggedRtpPacket(timestamp, parsed_header, total_length)); |
195 break; | 191 break; |
196 } | 192 } |
197 case ParsedRtcEventLog::RTCP_EVENT: { | 193 case ParsedRtcEventLog::RTCP_EVENT: { |
194 uint8_t packet[IP_PACKET_SIZE]; | |
195 MediaType media_type; | |
196 parsed_log_.GetRtcpPacket(i, &direction, &media_type, packet, | |
197 &total_length); | |
198 | |
199 RtpUtility::RtpHeaderParser rtp_parser(packet, total_length); | |
200 RTPHeader parsed_header; | |
201 RTC_CHECK(rtp_parser.ParseRtcp(&parsed_header)); | |
202 uint32_t ssrc = parsed_header.ssrc; | |
203 | |
204 RTCPUtility::RTCPParserV2 rtcp_parser(packet, total_length, true); | |
205 RTC_CHECK(rtcp_parser.IsValid()); | |
206 | |
207 RTCPUtility::RTCPPacketTypes packet_type = rtcp_parser.Begin(); | |
208 while (packet_type != RTCPUtility::RTCPPacketTypes::kInvalid) { | |
209 switch (packet_type) { | |
210 case RTCPUtility::RTCPPacketTypes::kTransportFeedback: { | |
211 // Currently feedback is logged twice, both for audio and video. | |
212 // Only act on one of them. | |
213 if (media_type == MediaType::VIDEO) { | |
214 std::unique_ptr<rtcp::RtcpPacket> rtcp_packet( | |
215 rtcp_parser.ReleaseRtcpPacket()); | |
216 StreamId stream(ssrc, direction); | |
217 uint64_t timestamp = parsed_log_.GetTimestamp(i); | |
218 rtcp_packets_[stream].push_back(LoggedRtcpPacket( | |
219 timestamp, kRtcpTransportFeedback, std::move(rtcp_packet))); | |
220 } | |
221 break; | |
222 } | |
223 default: | |
224 break; | |
225 } | |
226 rtcp_parser.Iterate(); | |
227 packet_type = rtcp_parser.PacketType(); | |
228 } | |
198 break; | 229 break; |
199 } | 230 } |
200 case ParsedRtcEventLog::LOG_START: { | 231 case ParsedRtcEventLog::LOG_START: { |
201 break; | 232 break; |
202 } | 233 } |
203 case ParsedRtcEventLog::LOG_END: { | 234 case ParsedRtcEventLog::LOG_END: { |
204 break; | 235 break; |
205 } | 236 } |
206 case ParsedRtcEventLog::BWE_PACKET_LOSS_EVENT: { | 237 case ParsedRtcEventLog::BWE_PACKET_LOSS_EVENT: { |
207 BwePacketLossEvent bwe_update; | 238 BwePacketLossEvent bwe_update; |
(...skipping 17 matching lines...) Expand all Loading... | |
225 } | 256 } |
226 | 257 |
227 if (last_timestamp < first_timestamp) { | 258 if (last_timestamp < first_timestamp) { |
228 // No useful events in the log. | 259 // No useful events in the log. |
229 first_timestamp = last_timestamp = 0; | 260 first_timestamp = last_timestamp = 0; |
230 } | 261 } |
231 begin_time_ = first_timestamp; | 262 begin_time_ = first_timestamp; |
232 end_time_ = last_timestamp; | 263 end_time_ = last_timestamp; |
233 } | 264 } |
234 | 265 |
266 class BitrateObserver : public CongestionController::Observer, | |
267 public RemoteBitrateObserver { | |
268 public: | |
269 BitrateObserver() : last_bitrate_bps_(0), bitrate_updated_(false) {} | |
270 | |
271 void OnNetworkChanged(uint32_t bitrate_bps, | |
272 uint8_t fraction_loss, | |
273 int64_t rtt_ms) override { | |
274 last_bitrate_bps_ = bitrate_bps; | |
275 bitrate_updated_ = true; | |
276 } | |
277 | |
278 void OnReceiveBitrateChanged(const std::vector<uint32_t>& ssrcs, | |
279 uint32_t bitrate) override {} | |
280 | |
281 uint32_t last_bitrate_bps() const { return last_bitrate_bps_; } | |
282 bool GetAndResetBitrateUpdated() { | |
283 bool bitrate_updated = bitrate_updated_; | |
284 bitrate_updated_ = false; | |
285 return bitrate_updated; | |
286 } | |
287 | |
288 private: | |
289 uint32_t last_bitrate_bps_; | |
290 bool bitrate_updated_; | |
291 }; | |
292 | |
235 void EventLogAnalyzer::CreatePacketGraph(PacketDirection desired_direction, | 293 void EventLogAnalyzer::CreatePacketGraph(PacketDirection desired_direction, |
236 Plot* plot) { | 294 Plot* plot) { |
237 std::map<uint32_t, TimeSeries> time_series; | 295 std::map<uint32_t, TimeSeries> time_series; |
238 | 296 |
239 PacketDirection direction; | 297 PacketDirection direction; |
240 MediaType media_type; | 298 MediaType media_type; |
241 uint8_t header[IP_PACKET_SIZE]; | 299 uint8_t header[IP_PACKET_SIZE]; |
242 size_t header_length, total_length; | 300 size_t header_length, total_length; |
243 float max_y = 0; | 301 float max_y = 0; |
244 | 302 |
(...skipping 423 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
668 plot->yaxis_min = kDefaultYMin; | 726 plot->yaxis_min = kDefaultYMin; |
669 plot->yaxis_max = max_y * kYMargin; | 727 plot->yaxis_max = max_y * kYMargin; |
670 plot->yaxis_label = "Bitrate (kbps)"; | 728 plot->yaxis_label = "Bitrate (kbps)"; |
671 if (desired_direction == webrtc::PacketDirection::kIncomingPacket) { | 729 if (desired_direction == webrtc::PacketDirection::kIncomingPacket) { |
672 plot->title = "Incoming bitrate per stream"; | 730 plot->title = "Incoming bitrate per stream"; |
673 } else if (desired_direction == webrtc::PacketDirection::kOutgoingPacket) { | 731 } else if (desired_direction == webrtc::PacketDirection::kOutgoingPacket) { |
674 plot->title = "Outgoing bitrate per stream"; | 732 plot->title = "Outgoing bitrate per stream"; |
675 } | 733 } |
676 } | 734 } |
677 | 735 |
736 void EventLogAnalyzer::CreateBweGraph(Plot* plot) { | |
737 std::map<uint64_t, const LoggedRtpPacket*> outgoing_rtp; | |
738 std::map<uint64_t, const LoggedRtcpPacket*> incoming_rtcp; | |
739 | |
740 for (const auto& kv : rtp_packets_) { | |
741 if (kv.first.GetDirection() == PacketDirection::kOutgoingPacket) { | |
742 for (const LoggedRtpPacket& rtp_packet : kv.second) | |
743 outgoing_rtp.insert(std::make_pair(rtp_packet.timestamp, &rtp_packet)); | |
744 } | |
745 } | |
746 | |
747 for (const auto& kv : rtcp_packets_) { | |
748 if (kv.first.GetDirection() == PacketDirection::kIncomingPacket) { | |
749 for (const LoggedRtcpPacket& rtcp_packet : kv.second) | |
750 incoming_rtcp.insert( | |
751 std::make_pair(rtcp_packet.timestamp, &rtcp_packet)); | |
752 } | |
753 } | |
754 | |
755 SimulatedClock clock(0); | |
756 BitrateObserver observer; | |
757 NullRtcEventLog null_event_log; | |
758 CongestionController cc(&clock, &observer, &observer, &null_event_log); | |
759 | |
760 TimeSeries time_series; | |
761 time_series.label = "BWE"; | |
762 time_series.style = LINE_DOT_GRAPH; | |
763 uint32_t max_y = 10; | |
764 uint32_t min_y = 0; | |
765 | |
766 auto rtp_iterator = outgoing_rtp.begin(); | |
767 auto rtcp_iterator = incoming_rtcp.begin(); | |
768 while (rtp_iterator != outgoing_rtp.end() || | |
769 rtcp_iterator != incoming_rtcp.end()) { | |
770 bool have_rtcp = rtcp_iterator != incoming_rtcp.end(); | |
771 bool have_rtp = rtp_iterator != outgoing_rtp.end(); | |
772 bool rtcp_next = false; | |
773 if (have_rtcp && have_rtp) { | |
774 rtcp_next = rtcp_iterator->first < rtp_iterator->first; | |
775 } else { | |
776 rtcp_next = have_rtcp; | |
777 } | |
778 | |
779 if (rtcp_next) { | |
780 clock.AdvanceTimeMilliseconds(rtcp_iterator->first / 1000 - | |
781 clock.TimeInMilliseconds()); | |
782 const LoggedRtcpPacket& rtcp = *rtcp_iterator->second; | |
783 if (rtcp.type == kRtcpTransportFeedback) { | |
784 cc.GetTransportFeedbackObserver()->OnTransportFeedback( | |
785 *static_cast<rtcp::TransportFeedback*>(rtcp.packet.get())); | |
terelius
2016/07/28 13:22:35
Is it safe to cast an RtcpPacket to a TransportFee
stefan-webrtc
2016/07/28 13:47:23
Yes, assuming the type is checked first (see line
| |
786 } | |
787 ++rtcp_iterator; | |
788 } else { | |
789 clock.AdvanceTimeMilliseconds(rtp_iterator->first / 1000 - | |
790 clock.TimeInMilliseconds()); | |
791 const LoggedRtpPacket& rtp = *rtp_iterator->second; | |
792 if (rtp.header.extension.hasTransportSequenceNumber) { | |
793 RTC_DCHECK(rtp.header.extension.hasTransportSequenceNumber); | |
794 cc.GetTransportFeedbackObserver()->AddPacket( | |
795 rtp.header.extension.transportSequenceNumber, rtp.total_length, 0); | |
796 rtc::SentPacket sent_packet( | |
797 rtp.header.extension.transportSequenceNumber, rtp.timestamp / 1000); | |
798 cc.OnSentPacket(sent_packet); | |
799 } | |
800 ++rtp_iterator; | |
801 } | |
802 if (cc.TimeUntilNextProcess() <= 0) | |
terelius
2016/07/28 13:22:35
Process is called at most once per packet in this
stefan-webrtc
2016/07/28 13:47:23
Agree, that's nicer, I'll do exactly like you sugg
| |
803 cc.Process(); | |
804 if (observer.GetAndResetBitrateUpdated()) { | |
805 uint32_t y = observer.last_bitrate_bps() / 1000; | |
806 max_y = std::max(max_y, y); | |
807 min_y = std::min(min_y, y); | |
808 float x = static_cast<float>(clock.TimeInMicroseconds() - begin_time_) / | |
809 1000000; | |
810 time_series.points.emplace_back(x, y); | |
811 } | |
812 } | |
813 // Add the data set to the plot. | |
814 plot->series.push_back(std::move(time_series)); | |
815 | |
816 plot->xaxis_min = kDefaultXMin; | |
817 plot->xaxis_max = (end_time_ - begin_time_) / 1000000 * kXMargin; | |
818 plot->xaxis_label = "Time (s)"; | |
819 plot->yaxis_min = min_y - (kYMargin - 1) / 2 * (max_y - min_y); | |
820 plot->yaxis_max = max_y + (kYMargin - 1) / 2 * (max_y - min_y); | |
821 plot->yaxis_label = "Bitrate (kbps)"; | |
822 plot->title = "BWE"; | |
823 } | |
824 | |
678 } // namespace plotting | 825 } // namespace plotting |
679 } // namespace webrtc | 826 } // namespace webrtc |
OLD | NEW |