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 <algorithm>
6
7 #include "remoting/base/util.h"
8 #include "testing/gtest/include/gtest/gtest.h"
9 #include "third_party/libyuv/include/libyuv/convert_from_argb.h"
10 #include "third_party/webrtc/modules/desktop_capture/desktop_geometry.h"
11
12 static const int kWidth = 32 ;
13 static const int kHeight = 24 ;
14 static const int kBytesPerPixel = 4;
15 static const int kYStride = kWidth;
16 static const int kUvStride = kWidth / 2;
17 static const int kRgbStride = kWidth * kBytesPerPixel;
18 static const uint32 kFillColor = 0xffffff;
19
20 namespace remoting {
21
22 class YuvToRgbTester {
23 public:
YuvToRgbTester()24 YuvToRgbTester() {
25 yuv_buffer_size_ = (kYStride + kUvStride) * kHeight;
26 yuv_buffer_.reset(new uint8[yuv_buffer_size_]);
27 yplane_ = yuv_buffer_.get();
28 uplane_ = yplane_ + (kYStride * kHeight);
29 vplane_ = uplane_ + (kUvStride * kHeight / 2);
30
31 rgb_buffer_size_ = kWidth * kHeight * kBytesPerPixel;
32 rgb_buffer_.reset(new uint8[rgb_buffer_size_]);
33
34 ResetYuvBuffer();
35 ResetRgbBuffer();
36 }
37
~YuvToRgbTester()38 ~YuvToRgbTester() {}
39
ResetYuvBuffer()40 void ResetYuvBuffer() {
41 memset(yuv_buffer_.get(), 0, yuv_buffer_size_);
42 }
43
ResetRgbBuffer()44 void ResetRgbBuffer() {
45 memset(rgb_buffer_.get(), 0, rgb_buffer_size_);
46 }
47
FillRgbBuffer(const webrtc::DesktopRect & rect)48 void FillRgbBuffer(const webrtc::DesktopRect& rect) {
49 uint32* ptr = reinterpret_cast<uint32*>(
50 rgb_buffer_.get() + (rect.top() * kRgbStride) +
51 (rect.left() * kBytesPerPixel));
52 int width = rect.width();
53 for (int height = rect.height(); height > 0; --height) {
54 std::fill(ptr, ptr + width, kFillColor);
55 ptr += kRgbStride / kBytesPerPixel;
56 }
57 }
58
59 // Check the the desination buffer is filled within expected bounds.
CheckRgbBuffer(const webrtc::DesktopRect & rect)60 void CheckRgbBuffer(const webrtc::DesktopRect& rect) {
61 uint32* ptr = reinterpret_cast<uint32*>(rgb_buffer_.get());
62 for (int y = 0; y < kHeight; ++y) {
63 if (y < rect.top() || rect.bottom() <= y) {
64 // The whole line should be intact.
65 EXPECT_EQ((ptrdiff_t)kWidth,
66 std::count(ptr, ptr + kWidth, 0u));
67 } else {
68 // The space before the painted rectangle should be intact.
69 EXPECT_EQ((ptrdiff_t)rect.left(),
70 std::count(ptr, ptr + rect.left(), 0u));
71
72 // All pixels of the target rectangle should be touched.
73 EXPECT_EQ(ptr + rect.right(),
74 std::find(ptr + rect.left(), ptr + rect.right(), 0u));
75
76 // The space after the painted rectangle should be intact.
77 EXPECT_EQ((ptrdiff_t)kWidth - rect.right(),
78 std::count(ptr + rect.right(), ptr + kWidth, 0u));
79 }
80 ptr += kRgbStride / kBytesPerPixel;
81 }
82 }
83
RunTest(const webrtc::DesktopSize dest_size,const webrtc::DesktopRect & rect)84 void RunTest(const webrtc::DesktopSize dest_size,
85 const webrtc::DesktopRect& rect) {
86 ASSERT_TRUE(
87 DoesRectContain(webrtc::DesktopRect::MakeSize(dest_size), rect));
88
89 // Reset buffers.
90 ResetYuvBuffer();
91 ResetRgbBuffer();
92 FillRgbBuffer(rect);
93
94 // RGB -> YUV
95 libyuv::ARGBToI420(rgb_buffer_.get(),
96 kRgbStride,
97 yplane_,
98 kYStride,
99 uplane_,
100 kUvStride,
101 vplane_,
102 kUvStride,
103 kWidth,
104 kHeight);
105
106 // Reset RGB buffer and do opposite conversion.
107 ResetRgbBuffer();
108 ConvertAndScaleYUVToRGB32Rect(yplane_,
109 uplane_,
110 vplane_,
111 kYStride,
112 kUvStride,
113 webrtc::DesktopSize(kWidth, kHeight),
114 webrtc::DesktopRect::MakeWH(kWidth, kHeight),
115 rgb_buffer_.get(),
116 kRgbStride,
117 dest_size,
118 webrtc::DesktopRect::MakeSize(dest_size),
119 rect);
120
121 // Check if it worked out.
122 CheckRgbBuffer(rect);
123 }
124
TestBasicConversion()125 void TestBasicConversion() {
126 // Whole buffer.
127 RunTest(webrtc::DesktopSize(kWidth, kHeight),
128 webrtc::DesktopRect::MakeWH(kWidth, kHeight));
129 }
130
131 private:
132 size_t yuv_buffer_size_;
133 scoped_ptr<uint8[]> yuv_buffer_;
134 uint8* yplane_;
135 uint8* uplane_;
136 uint8* vplane_;
137
138 size_t rgb_buffer_size_;
139 scoped_ptr<uint8[]> rgb_buffer_;
140
141 DISALLOW_COPY_AND_ASSIGN(YuvToRgbTester);
142 };
143
TEST(YuvToRgbTest,BasicConversion)144 TEST(YuvToRgbTest, BasicConversion) {
145 YuvToRgbTester tester;
146 tester.TestBasicConversion();
147 }
148
TEST(YuvToRgbTest,Clipping)149 TEST(YuvToRgbTest, Clipping) {
150 YuvToRgbTester tester;
151
152 webrtc::DesktopSize dest_size = webrtc::DesktopSize(kWidth, kHeight);
153 webrtc::DesktopRect rect =
154 webrtc::DesktopRect::MakeLTRB(0, 0, kWidth - 1, kHeight - 1);
155 // TODO(fbarchard): Allow top/left clipping to odd boundary.
156 for (int i = 0; i < 16; ++i) {
157 webrtc::DesktopRect dest_rect = webrtc::DesktopRect::MakeLTRB(
158 rect.left() + ((i & 1) ? 2 : 0),
159 rect.top() + ((i & 2) ? 2 : 0),
160 rect.right() - ((i & 4) ? 1 : 0),
161 rect.bottom() - ((i & 8) ? 1 : 0));
162
163 tester.RunTest(dest_size, dest_rect);
164 }
165 }
166
TEST(YuvToRgbTest,ClippingAndScaling)167 TEST(YuvToRgbTest, ClippingAndScaling) {
168 YuvToRgbTester tester;
169
170 webrtc::DesktopSize dest_size =
171 webrtc::DesktopSize(kWidth - 10, kHeight - 10);
172 webrtc::DesktopRect rect =
173 webrtc::DesktopRect::MakeLTRB(6, 6, kWidth - 11, kHeight - 11);
174 for (int i = 0; i < 16; ++i) {
175 webrtc::DesktopRect dest_rect = webrtc::DesktopRect::MakeLTRB(
176 rect.left() + ((i & 1) ? 2 : 0),
177 rect.top() + ((i & 2) ? 2 : 0),
178 rect.right() - ((i & 4) ? 1 : 0),
179 rect.bottom() - ((i & 8) ? 1 : 0));
180
181 tester.RunTest(dest_size, dest_rect);
182 }
183 }
184
TEST(ReplaceLfByCrLfTest,Basic)185 TEST(ReplaceLfByCrLfTest, Basic) {
186 EXPECT_EQ("ab", ReplaceLfByCrLf("ab"));
187 EXPECT_EQ("\r\nab", ReplaceLfByCrLf("\nab"));
188 EXPECT_EQ("\r\nab\r\n", ReplaceLfByCrLf("\nab\n"));
189 EXPECT_EQ("\r\nab\r\ncd", ReplaceLfByCrLf("\nab\ncd"));
190 EXPECT_EQ("\r\nab\r\ncd\r\n", ReplaceLfByCrLf("\nab\ncd\n"));
191 EXPECT_EQ("\r\n\r\nab\r\n\r\ncd\r\n\r\n",
192 ReplaceLfByCrLf("\n\nab\n\ncd\n\n"));
193 }
194
TEST(ReplaceLfByCrLfTest,Speed)195 TEST(ReplaceLfByCrLfTest, Speed) {
196 int kLineSize = 128;
197 std::string line(kLineSize, 'a');
198 line[kLineSize - 1] = '\n';
199 // Make a 10M string.
200 int kLineNum = 10 * 1024 * 1024 / kLineSize;
201 std::string buffer;
202 buffer.resize(kLineNum * kLineSize);
203 for (int i = 0; i < kLineNum; ++i) {
204 memcpy(&buffer[i * kLineSize], &line[0], kLineSize);
205 }
206 // Convert the string.
207 buffer = ReplaceLfByCrLf(buffer);
208 // Check the converted string.
209 EXPECT_EQ(static_cast<size_t>((kLineSize + 1) * kLineNum), buffer.size());
210 const char* p = &buffer[0];
211 for (int i = 0; i < kLineNum; ++i) {
212 EXPECT_EQ(0, memcmp(&line[0], p, kLineSize - 1));
213 p += kLineSize - 1;
214 EXPECT_EQ('\r', *p++);
215 EXPECT_EQ('\n', *p++);
216 }
217 }
218
TEST(ReplaceCrLfByLfTest,Basic)219 TEST(ReplaceCrLfByLfTest, Basic) {
220 EXPECT_EQ("ab", ReplaceCrLfByLf("ab"));
221 EXPECT_EQ("\nab", ReplaceCrLfByLf("\r\nab"));
222 EXPECT_EQ("\nab\n", ReplaceCrLfByLf("\r\nab\r\n"));
223 EXPECT_EQ("\nab\ncd", ReplaceCrLfByLf("\r\nab\r\ncd"));
224 EXPECT_EQ("\nab\ncd\n", ReplaceCrLfByLf("\r\nab\r\ncd\n"));
225 EXPECT_EQ("\n\nab\n\ncd\n\n",
226 ReplaceCrLfByLf("\r\n\r\nab\r\n\r\ncd\r\n\r\n"));
227 EXPECT_EQ("\rab\rcd\r", ReplaceCrLfByLf("\rab\rcd\r"));
228 }
229
TEST(ReplaceCrLfByLfTest,Speed)230 TEST(ReplaceCrLfByLfTest, Speed) {
231 int kLineSize = 128;
232 std::string line(kLineSize, 'a');
233 line[kLineSize - 2] = '\r';
234 line[kLineSize - 1] = '\n';
235 // Make a 10M string.
236 int kLineNum = 10 * 1024 * 1024 / kLineSize;
237 std::string buffer;
238 buffer.resize(kLineNum * kLineSize);
239 for (int i = 0; i < kLineNum; ++i) {
240 memcpy(&buffer[i * kLineSize], &line[0], kLineSize);
241 }
242 // Convert the string.
243 buffer = ReplaceCrLfByLf(buffer);
244 // Check the converted string.
245 EXPECT_EQ(static_cast<size_t>((kLineSize - 1) * kLineNum), buffer.size());
246 const char* p = &buffer[0];
247 for (int i = 0; i < kLineNum; ++i) {
248 EXPECT_EQ(0, memcmp(&line[0], p, kLineSize - 2));
249 p += kLineSize - 2;
250 EXPECT_EQ('\n', *p++);
251 }
252 }
253
TEST(StringIsUtf8Test,Basic)254 TEST(StringIsUtf8Test, Basic) {
255 EXPECT_TRUE(StringIsUtf8("", 0));
256 EXPECT_TRUE(StringIsUtf8("\0", 1));
257 EXPECT_TRUE(StringIsUtf8("abc", 3));
258 EXPECT_TRUE(StringIsUtf8("\xc0\x80", 2));
259 EXPECT_TRUE(StringIsUtf8("\xe0\x80\x80", 3));
260 EXPECT_TRUE(StringIsUtf8("\xf0\x80\x80\x80", 4));
261 EXPECT_TRUE(StringIsUtf8("\xf8\x80\x80\x80\x80", 5));
262 EXPECT_TRUE(StringIsUtf8("\xfc\x80\x80\x80\x80\x80", 6));
263
264 // Not enough continuation characters
265 EXPECT_FALSE(StringIsUtf8("\xc0", 1));
266 EXPECT_FALSE(StringIsUtf8("\xe0\x80", 2));
267 EXPECT_FALSE(StringIsUtf8("\xf0\x80\x80", 3));
268 EXPECT_FALSE(StringIsUtf8("\xf8\x80\x80\x80", 4));
269 EXPECT_FALSE(StringIsUtf8("\xfc\x80\x80\x80\x80", 5));
270
271 // One more continuation character than needed
272 EXPECT_FALSE(StringIsUtf8("\xc0\x80\x80", 3));
273 EXPECT_FALSE(StringIsUtf8("\xe0\x80\x80\x80", 4));
274 EXPECT_FALSE(StringIsUtf8("\xf0\x80\x80\x80\x80", 5));
275 EXPECT_FALSE(StringIsUtf8("\xf8\x80\x80\x80\x80\x80", 6));
276 EXPECT_FALSE(StringIsUtf8("\xfc\x80\x80\x80\x80\x80\x80", 7));
277
278 // Invalid first byte
279 EXPECT_FALSE(StringIsUtf8("\xfe\x80\x80\x80\x80\x80\x80", 7));
280 EXPECT_FALSE(StringIsUtf8("\xff\x80\x80\x80\x80\x80\x80", 7));
281
282 // Invalid continuation byte
283 EXPECT_FALSE(StringIsUtf8("\xc0\x00", 2));
284 EXPECT_FALSE(StringIsUtf8("\xc0\x40", 2));
285 EXPECT_FALSE(StringIsUtf8("\xc0\xc0", 2));
286 }
287
288 } // namespace remoting
289