OLD | NEW |
| (Empty) |
1 // Copyright 2016 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 "remoting/protocol/webrtc_frame_scheduler.h" | |
6 | |
7 #include <algorithm> | |
8 #include <memory> | |
9 | |
10 #include "base/location.h" | |
11 #include "base/logging.h" | |
12 #include "base/single_thread_task_runner.h" | |
13 #include "base/threading/thread_task_runner_handle.h" | |
14 #include "remoting/base/constants.h" | |
15 #include "remoting/proto/video.pb.h" | |
16 | |
17 namespace remoting { | |
18 namespace protocol { | |
19 | |
20 namespace { | |
21 | |
22 // Default target bitrate in kbps | |
23 const int kDefaultTargetBitrateKbps = 1000; | |
24 | |
25 } // namespace | |
26 | |
27 // The frame scheduler currently uses a simple polling technique | |
28 // at 30 FPS to capture, encode and send frames over webrtc transport. | |
29 // An improved solution will use target bitrate feedback to pace out | |
30 // the capture rate. | |
31 WebrtcFrameScheduler::WebrtcFrameScheduler( | |
32 scoped_refptr<base::SingleThreadTaskRunner> encode_task_runner, | |
33 std::unique_ptr<webrtc::DesktopCapturer> capturer, | |
34 WebrtcTransport* webrtc_transport, | |
35 std::unique_ptr<VideoEncoder> encoder) | |
36 : target_bitrate_kbps_(kDefaultTargetBitrateKbps), | |
37 main_task_runner_(base::ThreadTaskRunnerHandle::Get()), | |
38 encode_task_runner_(encode_task_runner), | |
39 capturer_(std::move(capturer)), | |
40 webrtc_transport_(webrtc_transport), | |
41 encoder_(std::move(encoder)), | |
42 weak_factory_(this) { | |
43 DCHECK(encode_task_runner_); | |
44 DCHECK(capturer_); | |
45 DCHECK(webrtc_transport_); | |
46 DCHECK(encoder_); | |
47 // Does not really start anything. Registers callback on this class. | |
48 capturer_->Start(this); | |
49 capture_timer_.reset(new base::RepeatingTimer()); | |
50 } | |
51 | |
52 WebrtcFrameScheduler::~WebrtcFrameScheduler() { | |
53 encode_task_runner_->DeleteSoon(FROM_HERE, encoder_.release()); | |
54 } | |
55 | |
56 void WebrtcFrameScheduler::Start() { | |
57 // Register for PLI requests. | |
58 webrtc_transport_->video_encoder_factory()->SetKeyFrameRequestCallback( | |
59 base::Bind(&WebrtcFrameScheduler::SetKeyFrameRequest, | |
60 base::Unretained(this))); | |
61 // Register for target bitrate notifications. | |
62 webrtc_transport_->video_encoder_factory()->SetTargetBitrateCallback( | |
63 base::Bind(&WebrtcFrameScheduler::SetTargetBitrate, | |
64 base::Unretained(this))); | |
65 } | |
66 | |
67 void WebrtcFrameScheduler::Stop() { | |
68 // Clear PLI request callback. | |
69 webrtc_transport_->video_encoder_factory()->SetKeyFrameRequestCallback( | |
70 base::Closure()); | |
71 webrtc_transport_->video_encoder_factory()->SetTargetBitrateCallback( | |
72 TargetBitrateCallback()); | |
73 // Cancel any pending encode. | |
74 task_tracker_.TryCancelAll(); | |
75 capture_timer_->Stop(); | |
76 } | |
77 | |
78 void WebrtcFrameScheduler::Pause(bool pause) { | |
79 if (pause) { | |
80 Stop(); | |
81 } else { | |
82 Start(); | |
83 } | |
84 } | |
85 | |
86 void WebrtcFrameScheduler::SetSizeCallback( | |
87 const VideoStream::SizeCallback& callback) { | |
88 size_callback_ = callback; | |
89 } | |
90 | |
91 void WebrtcFrameScheduler::SetKeyFrameRequest() { | |
92 VLOG(1) << "Request key frame"; | |
93 base::AutoLock lock(lock_); | |
94 key_frame_request_ = true; | |
95 if (!received_first_frame_request_) { | |
96 received_first_frame_request_ = true; | |
97 main_task_runner_->PostTask( | |
98 FROM_HERE, base::Bind(&WebrtcFrameScheduler::StartCaptureTimer, | |
99 weak_factory_.GetWeakPtr())); | |
100 } | |
101 } | |
102 | |
103 void WebrtcFrameScheduler::StartCaptureTimer() { | |
104 capture_timer_->Start(FROM_HERE, base::TimeDelta::FromSeconds(1) / 30, this, | |
105 &WebrtcFrameScheduler::CaptureNextFrame); | |
106 } | |
107 | |
108 void WebrtcFrameScheduler::SetTargetBitrate(int target_bitrate_kbps) { | |
109 VLOG(1) << "Set Target bitrate " << target_bitrate_kbps; | |
110 base::AutoLock lock(lock_); | |
111 target_bitrate_kbps_ = target_bitrate_kbps; | |
112 } | |
113 | |
114 bool WebrtcFrameScheduler::ClearAndGetKeyFrameRequest() { | |
115 base::AutoLock lock(lock_); | |
116 bool key_frame_request = key_frame_request_; | |
117 key_frame_request_ = false; | |
118 return key_frame_request; | |
119 } | |
120 | |
121 void WebrtcFrameScheduler::OnCaptureResult( | |
122 webrtc::DesktopCapturer::Result result, | |
123 std::unique_ptr<webrtc::DesktopFrame> frame) { | |
124 DCHECK(thread_checker_.CalledOnValidThread()); | |
125 | |
126 base::TimeTicks captured_ticks = base::TimeTicks::Now(); | |
127 int64_t capture_timestamp_ms = | |
128 (captured_ticks - base::TimeTicks()).InMilliseconds(); | |
129 capture_pending_ = false; | |
130 | |
131 // TODO(sergeyu): Handle ERROR_PERMANENT result here. | |
132 | |
133 if (encode_pending_) { | |
134 // TODO(isheriff): consider queuing here | |
135 VLOG(1) << "Dropping captured frame since encoder is still busy"; | |
136 return; | |
137 } | |
138 | |
139 last_capture_completed_ticks_ = captured_ticks; | |
140 | |
141 webrtc::DesktopVector dpi = | |
142 frame->dpi().is_zero() ? webrtc::DesktopVector(kDefaultDpi, kDefaultDpi) | |
143 : frame->dpi(); | |
144 | |
145 if (!frame_size_.equals(frame->size()) || !frame_dpi_.equals(dpi)) { | |
146 frame_size_ = frame->size(); | |
147 frame_dpi_ = dpi; | |
148 if (!size_callback_.is_null()) | |
149 size_callback_.Run(frame_size_, frame_dpi_); | |
150 } | |
151 encode_pending_ = true; | |
152 task_tracker_.PostTaskAndReplyWithResult( | |
153 encode_task_runner_.get(), FROM_HERE, | |
154 base::Bind(&WebrtcFrameScheduler::EncodeFrame, encoder_.get(), | |
155 base::Passed(&frame), target_bitrate_kbps_, | |
156 ClearAndGetKeyFrameRequest(), capture_timestamp_ms), | |
157 base::Bind(&WebrtcFrameScheduler::OnFrameEncoded, | |
158 weak_factory_.GetWeakPtr())); | |
159 } | |
160 | |
161 void WebrtcFrameScheduler::CaptureNextFrame() { | |
162 DCHECK(thread_checker_.CalledOnValidThread()); | |
163 | |
164 if (capture_pending_ || encode_pending_) { | |
165 VLOG(1) << "Capture/encode still pending.."; | |
166 return; | |
167 } | |
168 | |
169 capture_pending_ = true; | |
170 VLOG(1) << "Capture next frame after " | |
171 << (base::TimeTicks::Now() - last_capture_started_ticks_) | |
172 .InMilliseconds(); | |
173 last_capture_started_ticks_ = base::TimeTicks::Now(); | |
174 capturer_->Capture(webrtc::DesktopRegion()); | |
175 } | |
176 | |
177 // static | |
178 std::unique_ptr<VideoPacket> WebrtcFrameScheduler::EncodeFrame( | |
179 VideoEncoder* encoder, | |
180 std::unique_ptr<webrtc::DesktopFrame> frame, | |
181 uint32_t target_bitrate_kbps, | |
182 bool key_frame_request, | |
183 int64_t capture_time_ms) { | |
184 uint32_t flags = 0; | |
185 if (key_frame_request) | |
186 flags |= VideoEncoder::REQUEST_KEY_FRAME; | |
187 | |
188 base::TimeTicks current = base::TimeTicks::Now(); | |
189 encoder->UpdateTargetBitrate(target_bitrate_kbps); | |
190 std::unique_ptr<VideoPacket> packet = encoder->Encode(*frame, flags); | |
191 if (!packet) | |
192 return nullptr; | |
193 // TODO(isheriff): Note that while VideoPacket capture time is supposed | |
194 // to be capture duration, we (ab)use it for capture timestamp here. This | |
195 // will go away when we move away from VideoPacket. | |
196 packet->set_capture_time_ms(capture_time_ms); | |
197 | |
198 VLOG(1) << "Encode duration " | |
199 << (base::TimeTicks::Now() - current).InMilliseconds() | |
200 << " payload size " << packet->data().size(); | |
201 return packet; | |
202 } | |
203 | |
204 void WebrtcFrameScheduler::OnFrameEncoded(std::unique_ptr<VideoPacket> packet) { | |
205 DCHECK(thread_checker_.CalledOnValidThread()); | |
206 encode_pending_ = false; | |
207 if (!packet) | |
208 return; | |
209 base::TimeTicks current = base::TimeTicks::Now(); | |
210 float encoded_bits = packet->data().size() * 8.0; | |
211 | |
212 // Simplistic adaptation of frame polling in the range 5 FPS to 30 FPS. | |
213 uint32_t next_sched_ms = std::max( | |
214 33, std::min(static_cast<int>(encoded_bits / target_bitrate_kbps_), 200)); | |
215 if (webrtc_transport_->video_encoder_factory()->SendEncodedFrame( | |
216 std::move(packet)) >= 0) { | |
217 VLOG(1) << " Send duration " | |
218 << (base::TimeTicks::Now() - current).InMilliseconds() | |
219 << "next sched " << next_sched_ms; | |
220 } else { | |
221 LOG(ERROR) << "SendEncodedFrame() failed"; | |
222 } | |
223 capture_timer_->Start(FROM_HERE, | |
224 base::TimeDelta::FromMilliseconds(next_sched_ms), this, | |
225 &WebrtcFrameScheduler::CaptureNextFrame); | |
226 } | |
227 | |
228 } // namespace protocol | |
229 } // namespace remoting | |
OLD | NEW |