1 // Copyright (c) 2012 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 <list>
6
7 #include "base/compiler_specific.h"
8 #include "base/logging.h"
9 #include "base/memory/ref_counted.h"
10 #include "base/message_loop/message_loop.h"
11 #include "base/run_loop.h"
12 #include "remoting/host/desktop_resizer.h"
13 #include "remoting/host/resizing_host_observer.h"
14 #include "remoting/host/screen_resolution.h"
15 #include "testing/gtest/include/gtest/gtest.h"
16 #include "third_party/webrtc/modules/desktop_capture/desktop_geometry.h"
17
18 namespace remoting {
19
operator <<(std::ostream & os,const ScreenResolution & resolution)20 std::ostream& operator<<(std::ostream& os, const ScreenResolution& resolution) {
21 return os << resolution.dimensions().width() << "x"
22 << resolution.dimensions().height() << " @ "
23 << resolution.dpi().x() << "x" << resolution.dpi().y();
24 }
25
operator ==(const ScreenResolution & a,const ScreenResolution & b)26 bool operator==(const ScreenResolution& a, const ScreenResolution& b) {
27 return a.Equals(b);
28 }
29
30 const int kDefaultDPI = 96;
31
MakeResolution(int width,int height)32 ScreenResolution MakeResolution(int width, int height) {
33 return ScreenResolution(webrtc::DesktopSize(width, height),
34 webrtc::DesktopVector(kDefaultDPI, kDefaultDPI));
35 }
36
37 class FakeDesktopResizer : public DesktopResizer {
38 public:
FakeDesktopResizer(const ScreenResolution & initial_resolution,bool exact_size_supported,const ScreenResolution * supported_resolutions,int num_supported_resolutions)39 FakeDesktopResizer(const ScreenResolution& initial_resolution,
40 bool exact_size_supported,
41 const ScreenResolution* supported_resolutions,
42 int num_supported_resolutions)
43 : initial_resolution_(initial_resolution),
44 current_resolution_(initial_resolution),
45 exact_size_supported_(exact_size_supported),
46 set_resolution_call_count_(0) {
47 for (int i = 0; i < num_supported_resolutions; ++i) {
48 supported_resolutions_.push_back(supported_resolutions[i]);
49 }
50 }
51
~FakeDesktopResizer()52 virtual ~FakeDesktopResizer() {
53 EXPECT_EQ(initial_resolution_, GetCurrentResolution());
54 }
55
set_resolution_call_count()56 int set_resolution_call_count() { return set_resolution_call_count_; }
57
58 // remoting::DesktopResizer interface
GetCurrentResolution()59 virtual ScreenResolution GetCurrentResolution() OVERRIDE {
60 return current_resolution_;
61 }
GetSupportedResolutions(const ScreenResolution & preferred)62 virtual std::list<ScreenResolution> GetSupportedResolutions(
63 const ScreenResolution& preferred) OVERRIDE {
64 std::list<ScreenResolution> result = supported_resolutions_;
65 if (exact_size_supported_) {
66 result.push_back(preferred);
67 }
68 return result;
69 }
SetResolution(const ScreenResolution & resolution)70 virtual void SetResolution(const ScreenResolution& resolution) OVERRIDE {
71 current_resolution_ = resolution;
72 ++set_resolution_call_count_;
73 }
RestoreResolution(const ScreenResolution & resolution)74 virtual void RestoreResolution(const ScreenResolution& resolution) OVERRIDE {
75 current_resolution_ = resolution;
76 }
77
78 private:
79 ScreenResolution initial_resolution_;
80 ScreenResolution current_resolution_;
81 bool exact_size_supported_;
82 std::list<ScreenResolution> supported_resolutions_;
83
84 int set_resolution_call_count_;
85 };
86
87 class ResizingHostObserverTest : public testing::Test {
88 public:
ResizingHostObserverTest()89 ResizingHostObserverTest()
90 : desktop_resizer_(NULL),
91 now_(base::Time::Now()) {
92 }
93
94 // This needs to be public because the derived test-case class needs to
95 // pass it to Bind, which fails if it's protected.
GetTime()96 base::Time GetTime() {
97 return now_;
98 }
99
100 protected:
SetDesktopResizer(scoped_ptr<FakeDesktopResizer> desktop_resizer)101 void SetDesktopResizer(scoped_ptr<FakeDesktopResizer> desktop_resizer) {
102 CHECK(!desktop_resizer_) << "Call SetDeskopResizer once per test";
103 desktop_resizer_ = desktop_resizer.get();
104
105 resizing_host_observer_.reset(
106 new ResizingHostObserver(desktop_resizer.PassAs<DesktopResizer>()));
107 resizing_host_observer_->SetNowFunctionForTesting(
108 base::Bind(&ResizingHostObserverTest::GetTimeAndIncrement,
109 base::Unretained(this)));
110 }
111
GetBestResolution(const ScreenResolution & client_size)112 ScreenResolution GetBestResolution(const ScreenResolution& client_size) {
113 resizing_host_observer_->SetScreenResolution(client_size);
114 return desktop_resizer_->GetCurrentResolution();
115 }
116
VerifySizes(const ScreenResolution * client_sizes,const ScreenResolution * expected_sizes,int number_of_sizes)117 void VerifySizes(const ScreenResolution* client_sizes,
118 const ScreenResolution* expected_sizes,
119 int number_of_sizes) {
120 for (int i = 0; i < number_of_sizes; ++i) {
121 ScreenResolution best_size = GetBestResolution(client_sizes[i]);
122 EXPECT_EQ(expected_sizes[i], best_size)
123 << "Input resolution = " << client_sizes[i];
124 }
125 }
126
GetTimeAndIncrement()127 base::Time GetTimeAndIncrement() {
128 base::Time result = now_;
129 now_ += base::TimeDelta::FromSeconds(1);
130 return result;
131 }
132
133 scoped_ptr<ResizingHostObserver> resizing_host_observer_;
134 FakeDesktopResizer* desktop_resizer_;
135 base::Time now_;
136 };
137
138 // Check that the host is not resized if GetSupportedSizes returns an empty
139 // list (even if GetCurrentSize is supported).
TEST_F(ResizingHostObserverTest,EmptyGetSupportedSizes)140 TEST_F(ResizingHostObserverTest, EmptyGetSupportedSizes) {
141 ScreenResolution initial = MakeResolution(640, 480);
142 scoped_ptr<FakeDesktopResizer> desktop_resizer(
143 new FakeDesktopResizer(initial, false, NULL, 0));
144 SetDesktopResizer(desktop_resizer.Pass());
145
146 ScreenResolution client_sizes[] = { MakeResolution(200, 100),
147 MakeResolution(100, 200) };
148 ScreenResolution expected_sizes[] = { initial, initial };
149 VerifySizes(client_sizes, expected_sizes, arraysize(client_sizes));
150 }
151
152 // Check that if the implementation supports exact size matching, it is used.
TEST_F(ResizingHostObserverTest,SelectExactSize)153 TEST_F(ResizingHostObserverTest, SelectExactSize) {
154 scoped_ptr<FakeDesktopResizer> desktop_resizer(
155 new FakeDesktopResizer(MakeResolution(640, 480), true, NULL, 0));
156 SetDesktopResizer(desktop_resizer.Pass());
157
158 ScreenResolution client_sizes[] = { MakeResolution(200, 100),
159 MakeResolution(100, 200),
160 MakeResolution(640, 480),
161 MakeResolution(480, 640),
162 MakeResolution(1280, 1024) };
163 VerifySizes(client_sizes, client_sizes, arraysize(client_sizes));
164 }
165
166 // Check that if the implementation supports a size that is no larger than
167 // the requested size, then the largest such size is used.
TEST_F(ResizingHostObserverTest,SelectBestSmallerSize)168 TEST_F(ResizingHostObserverTest, SelectBestSmallerSize) {
169 ScreenResolution supported_sizes[] = { MakeResolution(639, 479),
170 MakeResolution(640, 480) };
171 scoped_ptr<FakeDesktopResizer> desktop_resizer(
172 new FakeDesktopResizer(MakeResolution(640, 480), false,
173 supported_sizes, arraysize(supported_sizes)));
174 SetDesktopResizer(desktop_resizer.Pass());
175
176 ScreenResolution client_sizes[] = { MakeResolution(639, 479),
177 MakeResolution(640, 480),
178 MakeResolution(641, 481),
179 MakeResolution(999, 999) };
180 ScreenResolution expected_sizes[] = { supported_sizes[0], supported_sizes[1],
181 supported_sizes[1], supported_sizes[1] };
182 VerifySizes(client_sizes, expected_sizes, arraysize(client_sizes));
183 }
184
185 // Check that if the implementation supports only sizes that are larger than
186 // the requested size, then the one that requires the least down-scaling.
TEST_F(ResizingHostObserverTest,SelectBestScaleFactor)187 TEST_F(ResizingHostObserverTest, SelectBestScaleFactor) {
188 ScreenResolution supported_sizes[] = { MakeResolution(100, 100),
189 MakeResolution(200, 100) };
190 scoped_ptr<FakeDesktopResizer> desktop_resizer(
191 new FakeDesktopResizer(MakeResolution(200, 100), false,
192 supported_sizes, arraysize(supported_sizes)));
193 SetDesktopResizer(desktop_resizer.Pass());
194
195 ScreenResolution client_sizes[] = { MakeResolution(1, 1),
196 MakeResolution(99, 99),
197 MakeResolution(199, 99) };
198 ScreenResolution expected_sizes[] = { supported_sizes[0], supported_sizes[0],
199 supported_sizes[1] };
200 VerifySizes(client_sizes, expected_sizes, arraysize(client_sizes));
201 }
202
203 // Check that if the implementation supports two sizes that have the same
204 // resultant scale factor, then the widest one is selected.
TEST_F(ResizingHostObserverTest,SelectWidest)205 TEST_F(ResizingHostObserverTest, SelectWidest) {
206 ScreenResolution supported_sizes[] = { MakeResolution(640, 480),
207 MakeResolution(480, 640) };
208 scoped_ptr<FakeDesktopResizer> desktop_resizer(
209 new FakeDesktopResizer(MakeResolution(480, 640), false,
210 supported_sizes, arraysize(supported_sizes)));
211 SetDesktopResizer(desktop_resizer.Pass());
212
213 ScreenResolution client_sizes[] = { MakeResolution(100, 100),
214 MakeResolution(480, 480),
215 MakeResolution(500, 500),
216 MakeResolution(640, 640),
217 MakeResolution(1000, 1000) };
218 ScreenResolution expected_sizes[] = { supported_sizes[0], supported_sizes[0],
219 supported_sizes[0], supported_sizes[0],
220 supported_sizes[0] };
221 VerifySizes(client_sizes, expected_sizes, arraysize(client_sizes));
222 }
223
224 // Check that if the best match for the client size doesn't change, then we
225 // don't call SetSize.
TEST_F(ResizingHostObserverTest,NoSetSizeForSameSize)226 TEST_F(ResizingHostObserverTest, NoSetSizeForSameSize) {
227 ScreenResolution supported_sizes[] = { MakeResolution(640, 480),
228 MakeResolution(480, 640) };
229 FakeDesktopResizer* desktop_resizer =
230 new FakeDesktopResizer(MakeResolution(640, 480), false,
231 supported_sizes, arraysize(supported_sizes));
232 SetDesktopResizer(scoped_ptr<FakeDesktopResizer>(desktop_resizer));
233
234 ScreenResolution client_sizes[] = { MakeResolution(640, 640),
235 MakeResolution(1024, 768),
236 MakeResolution(640, 480) };
237 ScreenResolution expected_sizes[] = { MakeResolution(640, 480),
238 MakeResolution(640, 480),
239 MakeResolution(640, 480) };
240 VerifySizes(client_sizes, expected_sizes, arraysize(client_sizes));
241 EXPECT_EQ(desktop_resizer->set_resolution_call_count(), 0);
242 }
243
244 // Check that desktop resizes are rate-limited, and that if multiple resize
245 // requests are received in the time-out period, the most recent is respected.
TEST_F(ResizingHostObserverTest,RateLimited)246 TEST_F(ResizingHostObserverTest, RateLimited) {
247 FakeDesktopResizer* desktop_resizer =
248 new FakeDesktopResizer(MakeResolution(640, 480), true, NULL, 0);
249 SetDesktopResizer(scoped_ptr<FakeDesktopResizer>(desktop_resizer));
250 resizing_host_observer_->SetNowFunctionForTesting(
251 base::Bind(&ResizingHostObserverTest::GetTime, base::Unretained(this)));
252
253 base::MessageLoop message_loop;
254 base::RunLoop run_loop;
255
256 EXPECT_EQ(GetBestResolution(MakeResolution(100, 100)),
257 MakeResolution(100, 100));
258 now_ += base::TimeDelta::FromMilliseconds(900);
259 EXPECT_EQ(GetBestResolution(MakeResolution(200, 200)),
260 MakeResolution(100, 100));
261 now_ += base::TimeDelta::FromMilliseconds(99);
262 EXPECT_EQ(GetBestResolution(MakeResolution(300, 300)),
263 MakeResolution(100, 100));
264 now_ += base::TimeDelta::FromMilliseconds(1);
265
266 // Due to the kMinimumResizeIntervalMs constant in resizing_host_observer.cc,
267 // We need to wait a total of 1000ms for the final resize to be processed.
268 // Since it was queued 900 + 99 ms after the first, we need to wait an
269 // additional 1ms. However, since RunLoop is not guaranteed to process tasks
270 // with the same due time in FIFO order, wait an additional 1ms for safety.
271 message_loop.PostDelayedTask(
272 FROM_HERE,
273 run_loop.QuitClosure(),
274 base::TimeDelta::FromMilliseconds(2));
275 run_loop.Run();
276
277 // If the QuitClosure fired before the final resize, it's a test failure.
278 EXPECT_EQ(desktop_resizer_->GetCurrentResolution(), MakeResolution(300, 300));
279 }
280
281 } // namespace remoting
282