Add ICU message format support
[chromium-blink-merge.git] / media / cast / sender / performance_metrics_overlay.cc
blob5b7c7147ee0d3d45c940905a6852b599a7cb1e7f
1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "media/cast/sender/performance_metrics_overlay.h"
7 #include <algorithm>
8 #include <string>
10 #include "base/logging.h"
11 #include "base/numerics/safe_conversions.h"
12 #include "base/strings/stringprintf.h"
13 #include "media/base/video_frame.h"
15 namespace media {
16 namespace cast {
18 namespace {
20 const int kScale = 4; // Physical pixels per one logical pixel.
21 const int kCharacterWidth = 3; // Logical pixel width of one character.
22 const int kCharacterHeight = 5; // Logical pixel height of one character.
23 const int kCharacterSpacing = 1; // Logical pixels between each character.
24 const int kLineSpacing = 2; // Logical pixels between each line of characters.
25 const int kPlane = 0; // Y-plane in YUV formats.
27 // For each pixel in the |rect| (logical coordinates), either decrease the
28 // intensity or increase it so that the resulting pixel has a perceivably
29 // different value than it did before. |p_ul| is a pointer to the pixel at
30 // coordinate (0,0) in a single-channel 8bpp bitmap. |stride| is the number of
31 // bytes per row in the output bitmap.
32 void DivergePixels(const gfx::Rect& rect, uint8* p_ul, int stride) {
33 DCHECK(p_ul);
34 DCHECK_GT(stride, 0);
36 // These constants and heuristics were chosen based on experimenting with a
37 // wide variety of content, and converging on a readable result. The amount
38 // by which the darker pixels are changed is less because each unit of change
39 // has a larger visual impact on the darker end of the spectrum. Each pixel's
40 // intensity value is changed as follows:
42 // [16,31] --> [32,63] (always a difference of +16)
43 // [32,64] --> 16 (a difference between -16 and -48)
44 // [65,235] --> [17,187] (always a difference of -48)
45 const int kDivergeDownThreshold = 32;
46 const int kDivergeDownAmount = 48;
47 const int kDivergeUpAmount = 32;
48 const int kMinIntensity = 16;
50 const int top = rect.y() * kScale;
51 const int bottom = rect.bottom() * kScale;
52 const int left = rect.x() * kScale;
53 const int right = rect.right() * kScale;
54 for (int y = top; y < bottom; ++y) {
55 uint8* const p_l = p_ul + y * stride;
56 for (int x = left; x < right; ++x) {
57 int intensity = p_l[x];
58 if (intensity >= kDivergeDownThreshold)
59 intensity = std::max(kMinIntensity, intensity - kDivergeDownAmount);
60 else
61 intensity += kDivergeUpAmount;
62 p_l[x] = static_cast<uint8>(intensity);
67 // Render |line| into |frame| at physical pixel row |top| and aligned to the
68 // right edge. Only number digits and a smattering of punctuation characters
69 // will be rendered.
70 void RenderLineOfText(const std::string& line, int top, VideoFrame* frame) {
71 // Compute number of physical pixels wide the rendered |line| would be,
72 // including padding.
73 const int line_width =
74 (((kCharacterWidth + kCharacterSpacing) * static_cast<int>(line.size())) +
75 kCharacterSpacing) * kScale;
77 // Determine if any characters would render past the left edge of the frame,
78 // and compute the index of the first character to be rendered.
79 const int pixels_per_char = (kCharacterWidth + kCharacterSpacing) * kScale;
80 const size_t first_idx = (line_width < frame->visible_rect().width()) ? 0u :
81 static_cast<size_t>(
82 ((line_width - frame->visible_rect().width()) / pixels_per_char) + 1);
84 // Compute the pointer to the pixel at the upper-left corner of the first
85 // character to be rendered.
86 const int stride = frame->stride(kPlane);
87 uint8* p_ul =
88 // Start at the first pixel in the first row...
89 frame->visible_data(kPlane) + (stride * top)
90 // ...now move to the right edge of the visible part of the frame...
91 + frame->visible_rect().width()
92 // ...now move left to where line[0] would be rendered...
93 - line_width
94 // ...now move right to where line[first_idx] would be rendered.
95 + first_idx * pixels_per_char;
97 // Render each character.
98 for (size_t i = first_idx; i < line.size(); ++i, p_ul += pixels_per_char) {
99 switch (line[i]) {
100 case '0':
101 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
102 DivergePixels(gfx::Rect(0, 1, 1, 3), p_ul, stride);
103 DivergePixels(gfx::Rect(2, 1, 1, 3), p_ul, stride);
104 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
105 break;
106 case '1':
107 DivergePixels(gfx::Rect(1, 0, 1, 5), p_ul, stride);
108 break;
109 case '2':
110 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
111 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
112 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
113 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
114 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
115 break;
116 case '3':
117 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
118 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
119 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
120 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
121 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
122 break;
123 case '4':
124 DivergePixels(gfx::Rect(0, 0, 1, 2), p_ul, stride);
125 DivergePixels(gfx::Rect(2, 0, 1, 5), p_ul, stride);
126 DivergePixels(gfx::Rect(0, 2, 2, 1), p_ul, stride);
127 break;
128 case '5':
129 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
130 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
131 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
132 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
133 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
134 break;
135 case '6':
136 DivergePixels(gfx::Rect(1, 0, 2, 1), p_ul, stride);
137 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
138 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
139 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
140 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
141 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
142 break;
143 case '7':
144 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
145 DivergePixels(gfx::Rect(2, 1, 1, 2), p_ul, stride);
146 DivergePixels(gfx::Rect(1, 3, 1, 2), p_ul, stride);
147 break;
148 case '8':
149 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
150 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
151 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
152 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
153 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
154 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
155 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
156 break;
157 case '9':
158 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
159 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
160 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
161 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
162 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
163 DivergePixels(gfx::Rect(0, 4, 2, 1), p_ul, stride);
164 break;
165 case 'e':
166 case 'E':
167 DivergePixels(gfx::Rect(0, 0, 3, 1), p_ul, stride);
168 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
169 DivergePixels(gfx::Rect(0, 2, 2, 1), p_ul, stride);
170 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
171 DivergePixels(gfx::Rect(0, 4, 3, 1), p_ul, stride);
172 break;
173 case '.':
174 DivergePixels(gfx::Rect(1, 4, 1, 1), p_ul, stride);
175 break;
176 case '+':
177 DivergePixels(gfx::Rect(1, 1, 1, 1), p_ul, stride);
178 DivergePixels(gfx::Rect(1, 3, 1, 1), p_ul, stride);
179 // ...fall through...
180 case '-':
181 DivergePixels(gfx::Rect(0, 2, 3, 1), p_ul, stride);
182 break;
183 case 'x':
184 DivergePixels(gfx::Rect(0, 1, 1, 1), p_ul, stride);
185 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
186 DivergePixels(gfx::Rect(1, 2, 1, 1), p_ul, stride);
187 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
188 DivergePixels(gfx::Rect(2, 3, 1, 1), p_ul, stride);
189 break;
190 case ':':
191 DivergePixels(gfx::Rect(1, 1, 1, 1), p_ul, stride);
192 DivergePixels(gfx::Rect(1, 3, 1, 1), p_ul, stride);
193 break;
194 case '%':
195 DivergePixels(gfx::Rect(0, 0, 1, 1), p_ul, stride);
196 DivergePixels(gfx::Rect(2, 1, 1, 1), p_ul, stride);
197 DivergePixels(gfx::Rect(1, 2, 1, 1), p_ul, stride);
198 DivergePixels(gfx::Rect(0, 3, 1, 1), p_ul, stride);
199 DivergePixels(gfx::Rect(2, 4, 1, 1), p_ul, stride);
200 break;
201 case ' ':
202 default:
203 break;
208 } // namespace
210 void MaybeRenderPerformanceMetricsOverlay(int target_bitrate,
211 int frames_ago,
212 double deadline_utilization,
213 double lossy_utilization,
214 VideoFrame* frame) {
215 if (VideoFrame::PlaneHorizontalBitsPerPixel(frame->format(), kPlane) != 8) {
216 DLOG(WARNING) << "Cannot render overlay: Plane " << kPlane << " not 8bpp.";
217 return;
220 // Compute the physical pixel top row for the bottom-most line of text.
221 const int line_height = (kCharacterHeight + kLineSpacing) * kScale;
222 int top = frame->visible_rect().height() - line_height;
223 if (top < 0 || !VLOG_IS_ON(1))
224 return;
226 // Line 3: Frame resolution and timestamp.
227 base::TimeDelta rem = frame->timestamp();
228 const int minutes = rem.InMinutes();
229 rem -= base::TimeDelta::FromMinutes(minutes);
230 const int seconds = static_cast<int>(rem.InSeconds());
231 rem -= base::TimeDelta::FromSeconds(seconds);
232 const int hundredth_seconds = static_cast<int>(rem.InMilliseconds() / 10);
233 RenderLineOfText(base::StringPrintf("%dx%d %d:%02d.%02d",
234 frame->visible_rect().width(),
235 frame->visible_rect().height(),
236 minutes,
237 seconds,
238 hundredth_seconds),
239 top,
240 frame);
242 // Move up one line's worth of pixels.
243 top -= line_height;
244 if (top < 0 || !VLOG_IS_ON(2))
245 return;
247 // Line 2: Capture/frame duration and target bitrate.
248 int capture_duration_ms = 0;
249 base::TimeTicks capture_begin_time, capture_end_time;
250 if (frame->metadata()->GetTimeTicks(VideoFrameMetadata::CAPTURE_BEGIN_TIME,
251 &capture_begin_time) &&
252 frame->metadata()->GetTimeTicks(VideoFrameMetadata::CAPTURE_END_TIME,
253 &capture_end_time)) {
254 capture_duration_ms = base::saturated_cast<int>(
255 (capture_end_time - capture_begin_time).InMillisecondsF() + 0.5);
257 int frame_duration_ms = 0;
258 int frame_duration_ms_frac = 0;
259 base::TimeDelta frame_duration;
260 if (frame->metadata()->GetTimeDelta(VideoFrameMetadata::FRAME_DURATION,
261 &frame_duration)) {
262 const int decimilliseconds = base::saturated_cast<int>(
263 frame_duration.InMicroseconds() / 100.0 + 0.5);
264 frame_duration_ms = decimilliseconds / 10;
265 frame_duration_ms_frac = decimilliseconds % 10;
267 const int target_kbits = target_bitrate / 1000;
268 RenderLineOfText(base::StringPrintf("%3.1d %3.1d.%01d %4.1d",
269 capture_duration_ms,
270 frame_duration_ms,
271 frame_duration_ms_frac,
272 target_kbits),
273 top,
274 frame);
276 // Move up one line's worth of pixels.
277 top -= line_height;
278 if (top < 0 || !VLOG_IS_ON(3))
279 return;
281 // Line 1: Recent utilization metrics.
282 const int deadline_pct =
283 base::saturated_cast<int>(deadline_utilization * 100.0 + 0.5);
284 const int lossy_pct =
285 base::saturated_cast<int>(lossy_utilization * 100.0 + 0.5);
286 RenderLineOfText(base::StringPrintf("%d %3.1d%% %3.1d%%",
287 frames_ago,
288 deadline_pct,
289 lossy_pct),
290 top,
291 frame);
294 } // namespace cast
295 } // namespace media