OLD | NEW |
1 /* | 1 /* |
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
(...skipping 19 matching lines...) Expand all Loading... |
30 captured_frame_.pixel_width = 1; | 30 captured_frame_.pixel_width = 1; |
31 captured_frame_.pixel_height = 1; | 31 captured_frame_.pixel_height = 1; |
32 captured_frame_.time_stamp = rtc::TimeNanos(); | 32 captured_frame_.time_stamp = rtc::TimeNanos(); |
33 captured_frame_.rotation = frame_rotation; | 33 captured_frame_.rotation = frame_rotation; |
34 captured_frame_.width = frame_width; | 34 captured_frame_.width = frame_width; |
35 captured_frame_.height = frame_height; | 35 captured_frame_.height = frame_height; |
36 captured_frame_.data_size = | 36 captured_frame_.data_size = |
37 (frame_width * frame_height) + | 37 (frame_width * frame_height) + |
38 ((frame_width + 1) / 2) * ((frame_height + 1) / 2) * 2; | 38 ((frame_width + 1) / 2) * ((frame_height + 1) / 2) * 2; |
39 captured_frame_buffer_.reset(new uint8_t[captured_frame_.data_size]); | 39 captured_frame_buffer_.reset(new uint8_t[captured_frame_.data_size]); |
40 // Initialize memory to satisfy DrMemory tests. | 40 |
41 memset(captured_frame_buffer_.get(), 0, captured_frame_.data_size); | 41 // Initialize with gradient, Y = 128(x/w + y/h), U = 256 x/w, V = 256 y/h |
| 42 for (int x = 0; x < frame_width; x++) |
| 43 for (int y = 0; y < frame_height; y++) { |
| 44 captured_frame_buffer_[x + y * frame_width] = |
| 45 128 * (x * frame_height + y * frame_width) / |
| 46 (frame_width * frame_height); |
| 47 } |
| 48 int chroma_width = (frame_width + 1) / 2; |
| 49 int chroma_height = (frame_height + 1) / 2; |
| 50 for (int x = 0; x < chroma_width; x++) |
| 51 for (int y = 0; y < chroma_height; y++) { |
| 52 captured_frame_buffer_[frame_width * frame_height + |
| 53 x + y*chroma_width] = 256 * x / chroma_width; |
| 54 captured_frame_buffer_[frame_width * frame_height + |
| 55 chroma_width * chroma_height + |
| 56 x + y*chroma_width] = 256 * y / chroma_height; |
| 57 } |
42 captured_frame_.data = captured_frame_buffer_.get(); | 58 captured_frame_.data = captured_frame_buffer_.get(); |
43 } | 59 } |
44 | 60 |
45 void VerifyFrame(cricket::VideoFrame* dest_frame, | 61 void VerifyFrame(cricket::VideoFrame* dest_frame, |
46 webrtc::VideoRotation src_rotation, | 62 webrtc::VideoRotation src_rotation, |
47 int src_width, | 63 int src_width, |
48 int src_height, | 64 int src_height, |
49 bool apply_rotation) { | 65 bool apply_rotation) { |
50 if (!apply_rotation) { | 66 if (!apply_rotation) { |
51 EXPECT_EQ(dest_frame->rotation(), src_rotation); | 67 EXPECT_EQ(dest_frame->rotation(), src_rotation); |
52 EXPECT_EQ(dest_frame->width(), src_width); | 68 EXPECT_EQ(dest_frame->width(), src_width); |
53 EXPECT_EQ(dest_frame->height(), src_height); | 69 EXPECT_EQ(dest_frame->height(), src_height); |
54 } else { | 70 } else { |
55 EXPECT_EQ(dest_frame->rotation(), webrtc::kVideoRotation_0); | 71 EXPECT_EQ(dest_frame->rotation(), webrtc::kVideoRotation_0); |
56 if (src_rotation == webrtc::kVideoRotation_90 || | 72 if (src_rotation == webrtc::kVideoRotation_90 || |
57 src_rotation == webrtc::kVideoRotation_270) { | 73 src_rotation == webrtc::kVideoRotation_270) { |
58 EXPECT_EQ(dest_frame->width(), src_height); | 74 EXPECT_EQ(dest_frame->width(), src_height); |
59 EXPECT_EQ(dest_frame->height(), src_width); | 75 EXPECT_EQ(dest_frame->height(), src_width); |
60 } else { | 76 } else { |
61 EXPECT_EQ(dest_frame->width(), src_width); | 77 EXPECT_EQ(dest_frame->width(), src_width); |
62 EXPECT_EQ(dest_frame->height(), src_height); | 78 EXPECT_EQ(dest_frame->height(), src_height); |
63 } | 79 } |
64 } | 80 } |
65 } | 81 } |
| 82 void VerifyCrop(webrtc::VideoFrameBuffer* frame, |
| 83 webrtc::VideoRotation src_rotation, |
| 84 double rel_width, |
| 85 double rel_height, |
| 86 bool apply_rotation) { |
| 87 // Corners, in counter-clockwise order. E.g, 90 degree rotation |
| 88 // maps (0,0) onto (1,0). |
| 89 struct { |
| 90 int x; |
| 91 int y; |
| 92 } corners[4] = { |
| 93 {0, 0}, {1, 0}, {1, 1}, {0, 1}, |
| 94 }; |
| 95 |
| 96 int rotation = apply_rotation ? src_rotation / 90 : 0; |
| 97 |
| 98 int width = frame->width(); |
| 99 int height = frame->height(); |
| 100 // Check that pixel values in the corners match the gradient used |
| 101 // for initialization. |
| 102 for (int i = 0; i < 4; i++) { |
| 103 // Pixel coordinates of corner 0, after rotation. |
| 104 int x = corners[(i + rotation) % 4].x * (width - 1); |
| 105 int y = corners[(i + rotation) % 4].y * (height - 1); |
| 106 // Pre-rotation coordinates, range 0.0 - 1.0 correspond to the |
| 107 // size of the uncropped input frame. |
| 108 double orig_x = (1 - rel_width) / 2 + corners[i].x * rel_width; |
| 109 double orig_y = (1 - rel_height) / 2 + corners[i].y * rel_height; |
| 110 |
| 111 EXPECT_NEAR(frame->DataY()[x + y * frame->StrideY()] / 256.0, |
| 112 (orig_x + orig_y) / 2, 0.01); |
| 113 EXPECT_NEAR(frame->DataU()[x / 2 + (y / 2) * frame->StrideU()] / 256.0, |
| 114 orig_x, 0.01); |
| 115 EXPECT_NEAR(frame->DataV()[x / 2 + (y / 2) * frame->StrideV()] / 256.0, |
| 116 orig_y, 0.01); |
| 117 } |
| 118 } |
66 | 119 |
67 void TestCreateAliasedFrame(bool apply_rotation) { | 120 void TestCreateAliasedFrame(bool apply_rotation) { |
68 cricket::VideoFrameFactory& factory = factory_; | 121 cricket::VideoFrameFactory& factory = factory_; |
69 factory.SetApplyRotation(apply_rotation); | 122 factory.SetApplyRotation(apply_rotation); |
70 InitFrame(webrtc::kVideoRotation_270); | 123 InitFrame(webrtc::kVideoRotation_270); |
71 const cricket::CapturedFrame& captured_frame = get_captured_frame(); | 124 const cricket::CapturedFrame& captured_frame = get_captured_frame(); |
72 // Create the new frame from the CapturedFrame. | 125 // Create the new frame from the CapturedFrame. |
73 std::unique_ptr<cricket::VideoFrame> frame; | 126 std::unique_ptr<cricket::VideoFrame> frame; |
74 int new_width = captured_frame.width / 2; | 127 int new_width = captured_frame.width / 2; |
75 int new_height = captured_frame.height / 2; | 128 int new_height = captured_frame.height / 2; |
76 frame.reset(factory.CreateAliasedFrame(&captured_frame, new_width, | 129 frame.reset(factory.CreateAliasedFrame(&captured_frame, new_width, |
77 new_height, new_width, new_height)); | 130 new_height, new_width, new_height)); |
78 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width, new_height, | 131 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width, new_height, |
79 apply_rotation); | 132 apply_rotation); |
80 | 133 |
81 frame.reset(factory.CreateAliasedFrame( | 134 frame.reset(factory.CreateAliasedFrame( |
82 &captured_frame, new_width, new_height, new_width / 2, new_height / 2)); | 135 &captured_frame, new_width, new_height, new_width / 2, new_height / 2)); |
83 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, | 136 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, |
84 new_height / 2, apply_rotation); | 137 new_height / 2, apply_rotation); |
85 | 138 |
86 // Reset the frame first so it's exclusive hence we could go through the | 139 // Reset the frame first so it's exclusive hence we could go through the |
87 // StretchToFrame code path in CreateAliasedFrame. | 140 // StretchToFrame code path in CreateAliasedFrame. |
88 frame.reset(); | 141 frame.reset(); |
89 frame.reset(factory.CreateAliasedFrame( | 142 frame.reset(factory.CreateAliasedFrame( |
90 &captured_frame, new_width, new_height, new_width / 2, new_height / 2)); | 143 &captured_frame, new_width, new_height, new_width / 2, new_height / 2)); |
91 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, | 144 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, |
92 new_height / 2, apply_rotation); | 145 new_height / 2, apply_rotation); |
93 } | 146 } |
94 | 147 |
| 148 void TestCreateScaledFrame(bool apply_rotation) { |
| 149 cricket::VideoFrameFactory& factory = factory_; |
| 150 factory.SetApplyRotation(apply_rotation); |
| 151 InitFrame(webrtc::kVideoRotation_270); |
| 152 const cricket::CapturedFrame& captured_frame = get_captured_frame(); |
| 153 // Create the new frame from the CapturedFrame. |
| 154 std::unique_ptr<cricket::VideoFrame> frame; |
| 155 int new_width = captured_frame.width / 2; |
| 156 int new_height = captured_frame.height / 2; |
| 157 frame = factory.CreateScaledFrame(&captured_frame, new_width, new_height); |
| 158 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width, new_height, |
| 159 apply_rotation); |
| 160 |
| 161 frame = factory.CreateScaledFrame(&captured_frame, |
| 162 new_width / 2, new_height / 2); |
| 163 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, |
| 164 new_height / 2, apply_rotation); |
| 165 |
| 166 // Vertical cropping |
| 167 frame = factory.CreateScaledFrame(&captured_frame, |
| 168 new_width / 2, new_height / 3); |
| 169 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 2, |
| 170 new_height / 3, apply_rotation); |
| 171 VerifyCrop(frame->video_frame_buffer(), webrtc::kVideoRotation_270, |
| 172 1.0, 2.0 / 3.0, apply_rotation); |
| 173 |
| 174 // Horizontal cropping |
| 175 frame = factory.CreateScaledFrame(&captured_frame, |
| 176 new_width / 3, new_height / 2); |
| 177 VerifyFrame(frame.get(), webrtc::kVideoRotation_270, new_width / 3, |
| 178 new_height / 2, apply_rotation); |
| 179 VerifyCrop(frame->video_frame_buffer(), webrtc::kVideoRotation_270, |
| 180 2.0 / 3.0, 1.0, apply_rotation); |
| 181 } |
95 const cricket::CapturedFrame& get_captured_frame() { return captured_frame_; } | 182 const cricket::CapturedFrame& get_captured_frame() { return captured_frame_; } |
96 | 183 |
97 private: | 184 private: |
98 cricket::CapturedFrame captured_frame_; | 185 cricket::CapturedFrame captured_frame_; |
99 std::unique_ptr<uint8_t[]> captured_frame_buffer_; | 186 std::unique_ptr<uint8_t[]> captured_frame_buffer_; |
100 cricket::WebRtcVideoFrameFactory factory_; | 187 cricket::WebRtcVideoFrameFactory factory_; |
101 }; | 188 }; |
102 | 189 |
103 TEST_F(WebRtcVideoFrameFactoryTest, NoApplyRotation) { | 190 TEST_F(WebRtcVideoFrameFactoryTest, CreateAliasedFrameNoApplyRotation) { |
104 TestCreateAliasedFrame(false); | 191 TestCreateAliasedFrame(false); |
105 } | 192 } |
106 | 193 |
107 TEST_F(WebRtcVideoFrameFactoryTest, ApplyRotation) { | 194 TEST_F(WebRtcVideoFrameFactoryTest, CreateAliasedFrameApplyRotation) { |
108 TestCreateAliasedFrame(true); | 195 TestCreateAliasedFrame(true); |
109 } | 196 } |
| 197 |
| 198 TEST_F(WebRtcVideoFrameFactoryTest, CreateScaledFrameNoApplyRotation) { |
| 199 TestCreateScaledFrame(false); |
| 200 } |
| 201 |
| 202 TEST_F(WebRtcVideoFrameFactoryTest, CreateScaledFrameApplyRotation) { |
| 203 TestCreateScaledFrame(true); |
| 204 } |
OLD | NEW |