Chromium Code Reviews

Side by Side Diff: content/browser/media/capture/video_capture_oracle.cc

Issue 1162863003: Move ContentVideoCaptureDeviceCore from src/content to src/media (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: rebase Created 5 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View unified diff |
OLDNEW
(Empty)
1 // Copyright (c) 2013 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.
4
5 #include "content/browser/media/capture/video_capture_oracle.h"
6
7 #include <algorithm>
8
9 #include "base/format_macros.h"
10 #include "base/strings/stringprintf.h"
11
12 namespace content {
13
14 namespace {
15
16 // This value controls how many redundant, timer-base captures occur when the
17 // content is static. Redundantly capturing the same frame allows iterative
18 // quality enhancement, and also allows the buffer to fill in "buffered mode".
19 //
20 // TODO(nick): Controlling this here is a hack and a layering violation, since
21 // it's a strategy specific to the WebRTC consumer, and probably just papers
22 // over some frame dropping and quality bugs. It should either be controlled at
23 // a higher level, or else redundant frame generation should be pushed down
24 // further into the WebRTC encoding stack.
25 const int kNumRedundantCapturesOfStaticContent = 200;
26
27 // Given the amount of time between frames, compare to the expected amount of
28 // time between frames at |frame_rate| and return the fractional difference.
29 double FractionFromExpectedFrameRate(base::TimeDelta delta, int frame_rate) {
30 DCHECK_GT(frame_rate, 0);
31 const base::TimeDelta expected_delta =
32 base::TimeDelta::FromSeconds(1) / frame_rate;
33 return (delta - expected_delta).InMillisecondsF() /
34 expected_delta.InMillisecondsF();
35 }
36
37 } // anonymous namespace
38
39 VideoCaptureOracle::VideoCaptureOracle(base::TimeDelta min_capture_period)
40 : next_frame_number_(0),
41 last_successfully_delivered_frame_number_(-1),
42 num_frames_pending_(0),
43 smoothing_sampler_(min_capture_period,
44 kNumRedundantCapturesOfStaticContent),
45 content_sampler_(min_capture_period) {
46 }
47
48 VideoCaptureOracle::~VideoCaptureOracle() {}
49
50 bool VideoCaptureOracle::ObserveEventAndDecideCapture(
51 Event event,
52 const gfx::Rect& damage_rect,
53 base::TimeTicks event_time) {
54 DCHECK_GE(event, 0);
55 DCHECK_LT(event, kNumEvents);
56 if (event_time < last_event_time_[event]) {
57 LOG(WARNING) << "Event time is not monotonically non-decreasing. "
58 << "Deciding not to capture this frame.";
59 return false;
60 }
61 last_event_time_[event] = event_time;
62
63 bool should_sample = false;
64 duration_of_next_frame_ = base::TimeDelta();
65 switch (event) {
66 case kCompositorUpdate:
67 smoothing_sampler_.ConsiderPresentationEvent(event_time);
68 content_sampler_.ConsiderPresentationEvent(damage_rect, event_time);
69 if (content_sampler_.HasProposal()) {
70 should_sample = content_sampler_.ShouldSample();
71 if (should_sample) {
72 event_time = content_sampler_.frame_timestamp();
73 duration_of_next_frame_ = content_sampler_.sampling_period();
74 }
75 } else {
76 should_sample = smoothing_sampler_.ShouldSample();
77 if (should_sample)
78 duration_of_next_frame_ = smoothing_sampler_.min_capture_period();
79 }
80 break;
81 case kTimerPoll:
82 // While the timer is firing, only allow a sampling if there are none
83 // currently in-progress.
84 if (num_frames_pending_ == 0) {
85 should_sample = smoothing_sampler_.IsOverdueForSamplingAt(event_time);
86 if (should_sample)
87 duration_of_next_frame_ = smoothing_sampler_.min_capture_period();
88 }
89 break;
90 case kNumEvents:
91 NOTREACHED();
92 break;
93 }
94
95 SetFrameTimestamp(next_frame_number_, event_time);
96 return should_sample;
97 }
98
99 int VideoCaptureOracle::RecordCapture() {
100 smoothing_sampler_.RecordSample();
101 content_sampler_.RecordSample(GetFrameTimestamp(next_frame_number_));
102 num_frames_pending_++;
103 return next_frame_number_++;
104 }
105
106 bool VideoCaptureOracle::CompleteCapture(int frame_number,
107 bool capture_was_successful,
108 base::TimeTicks* frame_timestamp) {
109 num_frames_pending_--;
110
111 // Drop frame if previously delivered frame number is higher.
112 if (last_successfully_delivered_frame_number_ > frame_number) {
113 LOG_IF(WARNING, capture_was_successful)
114 << "Out of order frame delivery detected (have #" << frame_number
115 << ", last was #" << last_successfully_delivered_frame_number_
116 << "). Dropping frame.";
117 return false;
118 }
119
120 if (!capture_was_successful) {
121 VLOG(2) << "Capture of frame #" << frame_number << " was not successful.";
122 return false;
123 }
124
125 DCHECK_NE(last_successfully_delivered_frame_number_, frame_number);
126 last_successfully_delivered_frame_number_ = frame_number;
127
128 *frame_timestamp = GetFrameTimestamp(frame_number);
129
130 // If enabled, log a measurement of how this frame timestamp has incremented
131 // in relation to an ideal increment.
132 if (VLOG_IS_ON(2) && frame_number > 0) {
133 const base::TimeDelta delta =
134 *frame_timestamp - GetFrameTimestamp(frame_number - 1);
135 if (content_sampler_.HasProposal()) {
136 const double estimated_frame_rate =
137 1000000.0 / content_sampler_.detected_period().InMicroseconds();
138 const int rounded_frame_rate =
139 static_cast<int>(estimated_frame_rate + 0.5);
140 VLOG(2) << base::StringPrintf(
141 "Captured #%d: delta=%" PRId64 " usec"
142 ", now locked into {%s}, %+0.1f%% slower than %d FPS",
143 frame_number,
144 delta.InMicroseconds(),
145 content_sampler_.detected_region().ToString().c_str(),
146 100.0 * FractionFromExpectedFrameRate(delta, rounded_frame_rate),
147 rounded_frame_rate);
148 } else {
149 VLOG(2) << base::StringPrintf(
150 "Captured #%d: delta=%" PRId64 " usec"
151 ", d/30fps=%+0.1f%%, d/25fps=%+0.1f%%, d/24fps=%+0.1f%%",
152 frame_number,
153 delta.InMicroseconds(),
154 100.0 * FractionFromExpectedFrameRate(delta, 30),
155 100.0 * FractionFromExpectedFrameRate(delta, 25),
156 100.0 * FractionFromExpectedFrameRate(delta, 24));
157 }
158 }
159
160 return !frame_timestamp->is_null();
161 }
162
163 base::TimeTicks VideoCaptureOracle::GetFrameTimestamp(int frame_number) const {
164 DCHECK_LE(frame_number, next_frame_number_);
165 DCHECK_LT(next_frame_number_ - frame_number, kMaxFrameTimestamps);
166 return frame_timestamps_[frame_number % kMaxFrameTimestamps];
167 }
168
169 void VideoCaptureOracle::SetFrameTimestamp(int frame_number,
170 base::TimeTicks timestamp) {
171 frame_timestamps_[frame_number % kMaxFrameTimestamps] = timestamp;
172 }
173
174 } // namespace content
OLDNEW
« no previous file with comments | « content/browser/media/capture/video_capture_oracle.h ('k') | content/browser/media/capture/video_capture_oracle_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine