OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2014 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 |
11 #include "webrtc/modules/video_coding/utility/quality_scaler.h" | 11 #include "webrtc/modules/video_coding/utility/quality_scaler.h" |
12 | 12 |
13 #include <memory> | 13 #include <memory> |
14 | 14 |
15 #include "webrtc/base/event.h" | 15 #include "webrtc/base/event.h" |
16 #include "webrtc/base/task_queue.h" | 16 #include "webrtc/base/task_queue.h" |
17 #include "webrtc/test/gmock.h" | 17 #include "webrtc/test/gmock.h" |
18 #include "webrtc/test/gtest.h" | 18 #include "webrtc/test/gtest.h" |
19 | 19 |
20 namespace webrtc { | 20 namespace webrtc { |
21 namespace { | 21 namespace { |
22 static const int kFramerate = 30; | 22 static const int kFramerate = 30; |
23 static const int kLowQp = 15; | 23 static const int kLowQp = 15; |
24 static const int kLowQpThreshold = 18; | 24 static const int kLowQpThreshold = 18; |
25 static const int kHighQp = 40; | 25 static const int kHighQp = 40; |
26 static const size_t kDefaultTimeoutMs = 150; | 26 static const size_t kDefaultTimeoutMs = 150; |
27 } // namespace | 27 } // namespace |
28 | 28 |
29 #define DO_SYNC(q, block) do { \ | |
30 rtc::Event event(false, false); \ | |
31 q->PostTask([this, &event] { \ | |
32 block; \ | |
33 event.Set(); \ | |
34 }); \ | |
35 event.Wait(1000); \ | |
sprang_webrtc
2017/03/29 08:53:05
RTC_CHECK this wait
| |
36 } while (0) | |
37 | |
sprang_webrtc
2017/03/26 12:24:47
I don't think this needs to be macro. Add a helper
kthelgason
2017/03/27 06:55:04
This can't be a function, at least with the curren
sprang_webrtc
2017/03/29 08:53:05
I don't like it, but fair enough :)
Let's see if t
| |
38 | |
29 class MockAdaptationObserver : public AdaptationObserverInterface { | 39 class MockAdaptationObserver : public AdaptationObserverInterface { |
30 public: | 40 public: |
31 MockAdaptationObserver() : event(false, false) {} | 41 MockAdaptationObserver() : event(false, false) {} |
32 virtual ~MockAdaptationObserver() {} | 42 virtual ~MockAdaptationObserver() {} |
33 | 43 |
34 void AdaptUp(AdaptReason r) override { | 44 void AdaptUp(AdaptReason r) override { |
35 adapt_up_events_++; | 45 adapt_up_events_++; |
36 event.Set(); | 46 event.Set(); |
37 } | 47 } |
38 void AdaptDown(AdaptReason r) override { | 48 void AdaptDown(AdaptReason r) override { |
(...skipping 19 matching lines...) Expand all Loading... | |
58 enum ScaleDirection { | 68 enum ScaleDirection { |
59 kKeepScaleAtHighQp, | 69 kKeepScaleAtHighQp, |
60 kScaleDown, | 70 kScaleDown, |
61 kScaleDownAboveHighQp, | 71 kScaleDownAboveHighQp, |
62 kScaleUp | 72 kScaleUp |
63 }; | 73 }; |
64 | 74 |
65 QualityScalerTest() | 75 QualityScalerTest() |
66 : q_(new rtc::TaskQueue("QualityScalerTestQueue")), | 76 : q_(new rtc::TaskQueue("QualityScalerTestQueue")), |
67 observer_(new MockAdaptationObserver()) { | 77 observer_(new MockAdaptationObserver()) { |
68 rtc::Event event(false, false); | 78 DO_SYNC(q_, { |
69 q_->PostTask([this, &event] { | |
70 qs_ = std::unique_ptr<QualityScaler>(new QualityScalerUnderTest( | 79 qs_ = std::unique_ptr<QualityScaler>(new QualityScalerUnderTest( |
71 observer_.get(), | 80 observer_.get(), |
72 VideoEncoder::QpThresholds(kLowQpThreshold, kHighQp))); | 81 VideoEncoder::QpThresholds(kLowQpThreshold, kHighQp)));}); |
73 event.Set(); | |
74 }); | |
75 EXPECT_TRUE(event.Wait(kDefaultTimeoutMs)); | |
76 } | 82 } |
77 | 83 |
78 ~QualityScalerTest() { | 84 ~QualityScalerTest() { |
79 rtc::Event event(false, false); | 85 DO_SYNC(q_, {qs_.reset(nullptr);}); |
80 q_->PostTask([this, &event] { | |
81 qs_.reset(nullptr); | |
82 event.Set(); | |
83 }); | |
84 EXPECT_TRUE(event.Wait(kDefaultTimeoutMs)); | |
85 } | 86 } |
86 | 87 |
87 void TriggerScale(ScaleDirection scale_direction) { | 88 void TriggerScale(ScaleDirection scale_direction) { |
88 for (int i = 0; i < kFramerate * 5; ++i) { | 89 for (int i = 0; i < kFramerate * 5; ++i) { |
89 switch (scale_direction) { | 90 switch (scale_direction) { |
90 case kScaleUp: | 91 case kScaleUp: |
91 qs_->ReportQP(kLowQp); | 92 qs_->ReportQP(kLowQp); |
92 break; | 93 break; |
93 case kScaleDown: | 94 case kScaleDown: |
94 qs_->ReportDroppedFrame(); | 95 qs_->ReportDroppedFrame(); |
95 break; | 96 break; |
96 case kKeepScaleAtHighQp: | 97 case kKeepScaleAtHighQp: |
97 qs_->ReportQP(kHighQp); | 98 qs_->ReportQP(kHighQp); |
98 break; | 99 break; |
99 case kScaleDownAboveHighQp: | 100 case kScaleDownAboveHighQp: |
100 qs_->ReportQP(kHighQp + 1); | 101 qs_->ReportQP(kHighQp + 1); |
101 break; | 102 break; |
102 } | 103 } |
103 } | 104 } |
104 } | 105 } |
105 | 106 |
106 std::unique_ptr<rtc::TaskQueue> q_; | 107 std::unique_ptr<rtc::TaskQueue> q_; |
107 std::unique_ptr<QualityScaler> qs_; | 108 std::unique_ptr<QualityScaler> qs_; |
108 std::unique_ptr<MockAdaptationObserver> observer_; | 109 std::unique_ptr<MockAdaptationObserver> observer_; |
109 }; | 110 }; |
110 | 111 |
111 #define DISABLED_TEST(basename, test) TEST_F(basename, DISABLED_##test) | 112 TEST_F(QualityScalerTest, DownscalesAfterContinuousFramedrop) { |
112 DISABLED_TEST(QualityScalerTest, DownscalesAfterContinuousFramedrop) { | 113 DO_SYNC(q_, { TriggerScale(kScaleDown); }); |
113 q_->PostTask([this] { TriggerScale(kScaleDown); }); | |
114 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 114 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
115 EXPECT_EQ(1, observer_->adapt_down_events_); | 115 EXPECT_EQ(1, observer_->adapt_down_events_); |
116 } | 116 } |
117 | 117 |
118 DISABLED_TEST(QualityScalerTest, KeepsScaleAtHighQp) { | 118 TEST_F(QualityScalerTest, KeepsScaleAtHighQp) { |
119 q_->PostTask([this] { TriggerScale(kKeepScaleAtHighQp); }); | 119 DO_SYNC(q_, { TriggerScale(kKeepScaleAtHighQp); }); |
120 EXPECT_FALSE(observer_->event.Wait(kDefaultTimeoutMs)); | 120 EXPECT_FALSE(observer_->event.Wait(kDefaultTimeoutMs)); |
121 EXPECT_EQ(0, observer_->adapt_down_events_); | 121 EXPECT_EQ(0, observer_->adapt_down_events_); |
122 EXPECT_EQ(0, observer_->adapt_up_events_); | 122 EXPECT_EQ(0, observer_->adapt_up_events_); |
123 } | 123 } |
124 | 124 |
125 DISABLED_TEST(QualityScalerTest, DownscalesAboveHighQp) { | 125 TEST_F(QualityScalerTest, DownscalesAboveHighQp) { |
126 q_->PostTask([this] { TriggerScale(kScaleDownAboveHighQp); }); | 126 DO_SYNC(q_, { TriggerScale(kScaleDownAboveHighQp); }); |
127 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 127 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
128 EXPECT_EQ(1, observer_->adapt_down_events_); | 128 EXPECT_EQ(1, observer_->adapt_down_events_); |
129 EXPECT_EQ(0, observer_->adapt_up_events_); | 129 EXPECT_EQ(0, observer_->adapt_up_events_); |
130 } | 130 } |
131 | 131 |
132 DISABLED_TEST(QualityScalerTest, DownscalesAfterTwoThirdsFramedrop) { | 132 TEST_F(QualityScalerTest, DownscalesAfterTwoThirdsFramedrop) { |
133 q_->PostTask([this] { | 133 DO_SYNC(q_, { |
134 qs_->ReportDroppedFrame(); | 134 qs_->ReportDroppedFrame(); |
135 qs_->ReportDroppedFrame(); | 135 qs_->ReportDroppedFrame(); |
136 qs_->ReportQP(kHighQp); | 136 qs_->ReportQP(kHighQp); |
137 }); | 137 }); |
138 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 138 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
139 EXPECT_EQ(1, observer_->adapt_down_events_); | 139 EXPECT_EQ(1, observer_->adapt_down_events_); |
140 EXPECT_EQ(0, observer_->adapt_up_events_); | 140 EXPECT_EQ(0, observer_->adapt_up_events_); |
141 } | 141 } |
142 | 142 |
143 DISABLED_TEST(QualityScalerTest, DoesNotDownscaleOnNormalQp) { | 143 TEST_F(QualityScalerTest, DoesNotDownscaleOnNormalQp) { |
144 q_->PostTask([this] { TriggerScale(kScaleDownAboveHighQp); }); | 144 DO_SYNC(q_, { TriggerScale(kScaleDownAboveHighQp); }); |
145 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 145 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
146 EXPECT_EQ(1, observer_->adapt_down_events_); | 146 EXPECT_EQ(1, observer_->adapt_down_events_); |
147 EXPECT_EQ(0, observer_->adapt_up_events_); | 147 EXPECT_EQ(0, observer_->adapt_up_events_); |
148 } | 148 } |
149 | 149 |
150 DISABLED_TEST(QualityScalerTest, DoesNotDownscaleAfterHalfFramedrop) { | 150 TEST_F(QualityScalerTest, DoesNotDownscaleAfterHalfFramedrop) { |
151 q_->PostTask([this] { | 151 DO_SYNC(q_, { |
152 qs_->ReportDroppedFrame(); | 152 qs_->ReportDroppedFrame(); |
153 qs_->ReportQP(kHighQp); | 153 qs_->ReportQP(kHighQp); |
154 }); | 154 }); |
155 EXPECT_FALSE(observer_->event.Wait(kDefaultTimeoutMs)); | 155 EXPECT_FALSE(observer_->event.Wait(kDefaultTimeoutMs)); |
156 EXPECT_EQ(0, observer_->adapt_down_events_); | 156 EXPECT_EQ(0, observer_->adapt_down_events_); |
157 EXPECT_EQ(0, observer_->adapt_up_events_); | 157 EXPECT_EQ(0, observer_->adapt_up_events_); |
158 } | 158 } |
159 | 159 |
160 DISABLED_TEST(QualityScalerTest, UpscalesAfterLowQp) { | 160 TEST_F(QualityScalerTest, UpscalesAfterLowQp) { |
161 q_->PostTask([this] { TriggerScale(kScaleUp); }); | 161 DO_SYNC(q_, { TriggerScale(kScaleUp); }); |
162 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 162 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
163 EXPECT_EQ(0, observer_->adapt_down_events_); | 163 EXPECT_EQ(0, observer_->adapt_down_events_); |
164 EXPECT_EQ(1, observer_->adapt_up_events_); | 164 EXPECT_EQ(1, observer_->adapt_up_events_); |
165 } | 165 } |
166 | 166 |
167 DISABLED_TEST(QualityScalerTest, ScalesDownAndBackUp) { | 167 TEST_F(QualityScalerTest, ScalesDownAndBackUp) { |
168 q_->PostTask([this] { TriggerScale(kScaleDown); }); | 168 DO_SYNC(q_, { TriggerScale(kScaleDown); }); |
169 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 169 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
170 EXPECT_EQ(1, observer_->adapt_down_events_); | 170 EXPECT_EQ(1, observer_->adapt_down_events_); |
171 EXPECT_EQ(0, observer_->adapt_up_events_); | 171 EXPECT_EQ(0, observer_->adapt_up_events_); |
172 q_->PostTask([this] { TriggerScale(kScaleUp); }); | 172 DO_SYNC(q_, { TriggerScale(kScaleUp); }); |
173 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); | 173 EXPECT_TRUE(observer_->event.Wait(kDefaultTimeoutMs)); |
174 EXPECT_EQ(1, observer_->adapt_down_events_); | 174 EXPECT_EQ(1, observer_->adapt_down_events_); |
175 EXPECT_EQ(1, observer_->adapt_up_events_); | 175 EXPECT_EQ(1, observer_->adapt_up_events_); |
176 } | 176 } |
177 #undef DISABLED_TEST | |
178 } // namespace webrtc | 177 } // namespace webrtc |
178 #undef DO_SYNC | |
OLD | NEW |