OLD | NEW |
(Empty) | |
| 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/skia/include/core/SkRect.h" |
| 10 #include "third_party/skia/include/core/SkSize.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: |
| 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 |
| 38 ~YuvToRgbTester() {} |
| 39 |
| 40 void ResetYuvBuffer() { |
| 41 memset(yuv_buffer_.get(), 0, yuv_buffer_size_); |
| 42 } |
| 43 |
| 44 void ResetRgbBuffer() { |
| 45 memset(rgb_buffer_.get(), 0, rgb_buffer_size_); |
| 46 } |
| 47 |
| 48 void FillRgbBuffer(const SkIRect& 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. |
| 60 void CheckRgbBuffer(const SkIRect& 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 |
| 84 void RunTest(const SkISize dest_size, const SkIRect& rect) { |
| 85 ASSERT_TRUE(SkIRect::MakeSize(dest_size).contains(rect)); |
| 86 |
| 87 // Reset buffers. |
| 88 ResetYuvBuffer(); |
| 89 ResetRgbBuffer(); |
| 90 FillRgbBuffer(rect); |
| 91 |
| 92 // RGB -> YUV |
| 93 ConvertRGB32ToYUVWithRect(rgb_buffer_.get(), |
| 94 yplane_, |
| 95 uplane_, |
| 96 vplane_, |
| 97 0, |
| 98 0, |
| 99 kWidth, |
| 100 kHeight, |
| 101 kRgbStride, |
| 102 kYStride, |
| 103 kUvStride); |
| 104 |
| 105 // Reset RGB buffer and do opposite conversion. |
| 106 ResetRgbBuffer(); |
| 107 ConvertAndScaleYUVToRGB32Rect(yplane_, |
| 108 uplane_, |
| 109 vplane_, |
| 110 kYStride, |
| 111 kUvStride, |
| 112 SkISize::Make(kWidth, kHeight), |
| 113 SkIRect::MakeWH(kWidth, kHeight), |
| 114 rgb_buffer_.get(), |
| 115 kRgbStride, |
| 116 dest_size, |
| 117 SkIRect::MakeSize(dest_size), |
| 118 rect); |
| 119 |
| 120 // Check if it worked out. |
| 121 CheckRgbBuffer(rect); |
| 122 } |
| 123 |
| 124 void TestBasicConversion() { |
| 125 // Whole buffer. |
| 126 RunTest(SkISize::Make(kWidth, kHeight), SkIRect::MakeWH(kWidth, kHeight)); |
| 127 } |
| 128 |
| 129 private: |
| 130 size_t yuv_buffer_size_; |
| 131 scoped_array<uint8> yuv_buffer_; |
| 132 uint8* yplane_; |
| 133 uint8* uplane_; |
| 134 uint8* vplane_; |
| 135 |
| 136 size_t rgb_buffer_size_; |
| 137 scoped_array<uint8> rgb_buffer_; |
| 138 |
| 139 DISALLOW_COPY_AND_ASSIGN(YuvToRgbTester); |
| 140 }; |
| 141 |
| 142 TEST(YuvToRgbTest, BasicConversion) { |
| 143 YuvToRgbTester tester; |
| 144 tester.TestBasicConversion(); |
| 145 } |
| 146 |
| 147 TEST(YuvToRgbTest, Clipping) { |
| 148 YuvToRgbTester tester; |
| 149 |
| 150 SkISize dest_size = SkISize::Make(kWidth, kHeight); |
| 151 SkIRect rect = SkIRect::MakeLTRB(0, 0, kWidth - 1, kHeight - 1); |
| 152 for (int i = 0; i < 16; ++i) { |
| 153 SkIRect dest_rect = rect; |
| 154 if ((i & 1) != 0) |
| 155 dest_rect.fLeft += 1; |
| 156 if ((i & 2) != 0) |
| 157 dest_rect.fTop += 1; |
| 158 if ((i & 4) != 0) |
| 159 dest_rect.fRight += 1; |
| 160 if ((i & 8) != 0) |
| 161 dest_rect.fBottom += 1; |
| 162 |
| 163 tester.RunTest(dest_size, dest_rect); |
| 164 } |
| 165 } |
| 166 |
| 167 TEST(YuvToRgbTest, ClippingAndScaling) { |
| 168 YuvToRgbTester tester; |
| 169 |
| 170 SkISize dest_size = SkISize::Make(kWidth - 10, kHeight - 10); |
| 171 SkIRect rect = SkIRect::MakeLTRB(5, 5, kWidth - 11, kHeight - 11); |
| 172 for (int i = 0; i < 16; ++i) { |
| 173 SkIRect dest_rect = rect; |
| 174 if ((i & 1) != 0) |
| 175 dest_rect.fLeft += 1; |
| 176 if ((i & 2) != 0) |
| 177 dest_rect.fTop += 1; |
| 178 if ((i & 4) != 0) |
| 179 dest_rect.fRight += 1; |
| 180 if ((i & 8) != 0) |
| 181 dest_rect.fBottom += 1; |
| 182 |
| 183 tester.RunTest(dest_size, dest_rect); |
| 184 } |
| 185 } |
| 186 |
| 187 } // namespace remoting |
OLD | NEW |