blob: bccedc85b5dff0c281d36f9e63b6c16e35d08806 [file] [log] [blame]
John Koleszarb9180fc2012-05-16 15:27:00 -07001/*
Yaowu Xu2ab7ff02016-09-02 12:04:54 -07002 * Copyright (c) 2016, Alliance for Open Media. All rights reserved
John Koleszarb9180fc2012-05-16 15:27:00 -07003 *
Yaowu Xu2ab7ff02016-09-02 12:04:54 -07004 * This source code is subject to the terms of the BSD 2 Clause License and
5 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6 * was not distributed with this source code in the LICENSE file, you can
7 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8 * Media Patent License 1.0 was not distributed with this source code in the
9 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
Johann123e8a62017-12-28 14:40:49 -080010 */
Deb Mukherjee01cafaa2013-01-15 06:43:35 -080011
Frank Galligan89df6d12015-03-09 15:52:29 -070012#include <string>
13
Tom Finegan7a07ece2017-02-07 17:14:05 -080014#include "third_party/googletest/src/googletest/include/gtest/gtest.h"
Jingning Han097d59c2015-07-29 14:51:36 -070015
Yaowu Xuf883b422016-08-30 14:01:10 -070016#include "./aom_config.h"
Yaowu Xuc27fc142016-08-22 16:08:15 -070017#include "aom_ports/mem.h"
John Koleszar706cafe2013-01-18 11:51:12 -080018#include "test/codec_factory.h"
Yaowu Xuc953aea2012-08-30 13:43:15 -070019#include "test/decode_test_driver.h"
Jingning Han097d59c2015-07-29 14:51:36 -070020#include "test/encode_test_driver.h"
James Zerneebb6482012-11-27 13:08:05 -080021#include "test/register_state_check.h"
John Koleszarb9180fc2012-05-16 15:27:00 -070022#include "test/video_source.h"
John Koleszarb9180fc2012-05-16 15:27:00 -070023
Yaowu Xuc27fc142016-08-22 16:08:15 -070024namespace libaom_test {
Yunqing Wang36664782014-12-12 14:34:30 -080025void Encoder::InitEncoder(VideoSource *video) {
Yaowu Xuf883b422016-08-30 14:01:10 -070026 aom_codec_err_t res;
27 const aom_image_t *img = video->img();
Yunqing Wang36664782014-12-12 14:34:30 -080028
29 if (video->img() && !encoder_.priv) {
30 cfg_.g_w = img->d_w;
31 cfg_.g_h = img->d_h;
32 cfg_.g_timebase = video->timebase();
33 cfg_.rc_twopass_stats_in = stats_->buf();
Frank Galligan89df6d12015-03-09 15:52:29 -070034
Yaowu Xuf883b422016-08-30 14:01:10 -070035 res = aom_codec_enc_init(&encoder_, CodecInterface(), &cfg_, init_flags_);
36 ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
Frank Galligan89df6d12015-03-09 15:52:29 -070037
Yaowu Xuf883b422016-08-30 14:01:10 -070038#if CONFIG_AV1_ENCODER
39 if (CodecInterface() == &aom_codec_av1_cx_algo) {
40// Default to 1 tile column for AV1. With CONFIG_EXT_TILE, the
clang-format3a826f12016-08-11 17:46:05 -070041// default is already the largest possible tile size
Geza Lore67a2ff72016-05-03 11:53:55 +010042#if !CONFIG_EXT_TILE
Jingning Han41be09a2015-08-19 14:13:18 -070043 const int log2_tile_columns = 0;
Yaowu Xuf883b422016-08-30 14:01:10 -070044 res = aom_codec_control_(&encoder_, AV1E_SET_TILE_COLUMNS,
Jingning Han41be09a2015-08-19 14:13:18 -070045 log2_tile_columns);
Yaowu Xuf883b422016-08-30 14:01:10 -070046 ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
Geza Lore67a2ff72016-05-03 11:53:55 +010047#endif // !CONFIG_EXT_TILE
Jingning Han41be09a2015-08-19 14:13:18 -070048 } else
49#endif
Yaowu Xu0924bcd2016-05-20 11:33:07 -070050 {
Frank Galligan89df6d12015-03-09 15:52:29 -070051 }
Yunqing Wang36664782014-12-12 14:34:30 -080052 }
53}
54
Adrian Grange4206c6d2012-10-02 11:03:09 -070055void Encoder::EncodeFrame(VideoSource *video, const unsigned long frame_flags) {
John Koleszarb9180fc2012-05-16 15:27:00 -070056 if (video->img())
Adrian Grange4206c6d2012-10-02 11:03:09 -070057 EncodeFrameInternal(*video, frame_flags);
John Koleszarb9180fc2012-05-16 15:27:00 -070058 else
59 Flush();
60
61 // Handle twopass stats
62 CxDataIterator iter = GetCxData();
63
Yaowu Xuf883b422016-08-30 14:01:10 -070064 while (const aom_codec_cx_pkt_t *pkt = iter.Next()) {
65 if (pkt->kind != AOM_CODEC_STATS_PKT) continue;
John Koleszarb9180fc2012-05-16 15:27:00 -070066
67 stats_->Append(*pkt);
68 }
69}
70
71void Encoder::EncodeFrameInternal(const VideoSource &video,
Adrian Grange4206c6d2012-10-02 11:03:09 -070072 const unsigned long frame_flags) {
Yaowu Xuf883b422016-08-30 14:01:10 -070073 aom_codec_err_t res;
74 const aom_image_t *img = video.img();
John Koleszarb9180fc2012-05-16 15:27:00 -070075
John Koleszarb9180fc2012-05-16 15:27:00 -070076 // Handle frame resizing
77 if (cfg_.g_w != img->d_w || cfg_.g_h != img->d_h) {
78 cfg_.g_w = img->d_w;
79 cfg_.g_h = img->d_h;
Yaowu Xuf883b422016-08-30 14:01:10 -070080 res = aom_codec_enc_config_set(&encoder_, &cfg_);
81 ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
John Koleszarb9180fc2012-05-16 15:27:00 -070082 }
83
84 // Encode the frame
Sean DuBois47cc2552018-01-23 07:44:16 +000085 API_REGISTER_STATE_CHECK(res =
86 aom_codec_encode(&encoder_, img, video.pts(),
87 video.duration(), frame_flags));
Yaowu Xuf883b422016-08-30 14:01:10 -070088 ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
John Koleszarb9180fc2012-05-16 15:27:00 -070089}
90
91void Encoder::Flush() {
Sean DuBois47cc2552018-01-23 07:44:16 +000092 const aom_codec_err_t res = aom_codec_encode(&encoder_, NULL, 0, 0, 0);
Deb Mukherjeebdbaa5b2014-07-18 03:06:07 -070093 if (!encoder_.priv)
Yaowu Xuf883b422016-08-30 14:01:10 -070094 ASSERT_EQ(AOM_CODEC_ERROR, res) << EncoderError();
Deb Mukherjeebdbaa5b2014-07-18 03:06:07 -070095 else
Yaowu Xuf883b422016-08-30 14:01:10 -070096 ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
John Koleszarb9180fc2012-05-16 15:27:00 -070097}
98
John Koleszar706cafe2013-01-18 11:51:12 -080099void EncoderTest::InitializeConfig() {
Yaowu Xuf883b422016-08-30 14:01:10 -0700100 const aom_codec_err_t res = codec_->DefaultEncoderConfig(&cfg_, 0);
Yaowu Xuf883b422016-08-30 14:01:10 -0700101 ASSERT_EQ(AOM_CODEC_OK, res);
John Koleszar706cafe2013-01-18 11:51:12 -0800102}
103
John Koleszarb9180fc2012-05-16 15:27:00 -0700104void EncoderTest::SetMode(TestMode mode) {
105 switch (mode) {
John Koleszarb9180fc2012-05-16 15:27:00 -0700106 case kOnePassGood:
Sean DuBois47cc2552018-01-23 07:44:16 +0000107 case kTwoPassGood: break;
108 case kRealTime: cfg_.g_lag_in_frames = 0; break;
clang-format3a826f12016-08-11 17:46:05 -0700109 default: ASSERT_TRUE(false) << "Unexpected mode " << mode;
John Koleszarb9180fc2012-05-16 15:27:00 -0700110 }
Thomas Daede80826142017-03-20 15:44:24 -0700111 mode_ = mode;
Thomas Daede6eca8352017-03-17 14:14:12 -0700112 if (mode == kTwoPassGood)
John Koleszarb9180fc2012-05-16 15:27:00 -0700113 passes_ = 2;
114 else
115 passes_ = 1;
116}
Geza Loree0dcab92016-05-09 13:19:34 +0100117
Yaowu Xu4ff59b52017-04-24 12:41:56 -0700118static bool compare_plane(const uint8_t *const buf1, int stride1,
119 const uint8_t *const buf2, int stride2, int w, int h,
120 int *const mismatch_row, int *const mismatch_col,
121 int *const mismatch_pix1, int *const mismatch_pix2) {
Geza Loree0dcab92016-05-09 13:19:34 +0100122 int r, c;
123
124 for (r = 0; r < h; ++r) {
125 for (c = 0; c < w; ++c) {
126 const int pix1 = buf1[r * stride1 + c];
127 const int pix2 = buf2[r * stride2 + c];
128
129 if (pix1 != pix2) {
clang-format3a826f12016-08-11 17:46:05 -0700130 if (mismatch_row != NULL) *mismatch_row = r;
131 if (mismatch_col != NULL) *mismatch_col = c;
132 if (mismatch_pix1 != NULL) *mismatch_pix1 = pix1;
133 if (mismatch_pix2 != NULL) *mismatch_pix2 = pix2;
Geza Loree0dcab92016-05-09 13:19:34 +0100134 return false;
135 }
136 }
137 }
138
139 return true;
140}
141
Yaowu Xuc953aea2012-08-30 13:43:15 -0700142// The function should return "true" most of the time, therefore no early
143// break-out is implemented within the match checking process.
Yaowu Xuf883b422016-08-30 14:01:10 -0700144static bool compare_img(const aom_image_t *img1, const aom_image_t *img2,
clang-format3a826f12016-08-11 17:46:05 -0700145 int *const mismatch_row, int *const mismatch_col,
146 int *const mismatch_plane, int *const mismatch_pix1,
Geza Loree0dcab92016-05-09 13:19:34 +0100147 int *const mismatch_pix2) {
Andrey Norkin9e694632017-12-21 18:50:57 -0800148#if CONFIG_CICP
149 if (img1->fmt != img2->fmt || img1->cp != img2->cp || img1->tc != img2->tc ||
150 img1->mc != img2->mc || img1->d_w != img2->d_w ||
Imdad Sardharwalla26ac0472018-01-18 15:00:24 +0000151 img1->d_h != img2->d_h || img1->monochrome != img2->monochrome) {
Andrey Norkin9e694632017-12-21 18:50:57 -0800152#else
clang-format3a826f12016-08-11 17:46:05 -0700153 if (img1->fmt != img2->fmt || img1->cs != img2->cs ||
Imdad Sardharwalla26ac0472018-01-18 15:00:24 +0000154 img1->d_w != img2->d_w || img1->d_h != img2->d_h ||
155 img1->monochrome != img2->monochrome) {
Andrey Norkin9e694632017-12-21 18:50:57 -0800156#endif
clang-format3a826f12016-08-11 17:46:05 -0700157 if (mismatch_row != NULL) *mismatch_row = -1;
158 if (mismatch_col != NULL) *mismatch_col = -1;
Geza Loree0dcab92016-05-09 13:19:34 +0100159 return false;
160 }
161
Imdad Sardharwalla26ac0472018-01-18 15:00:24 +0000162 const int num_planes = img1->monochrome ? 1 : 3;
163 for (int plane = 0; plane < num_planes; plane++) {
Sebastien Alaiwancdcf4b02017-09-21 11:29:47 +0200164 if (!compare_plane(img1->planes[plane], img1->stride[plane],
165 img2->planes[plane], img2->stride[plane],
166 aom_img_plane_width(img1, plane),
167 aom_img_plane_height(img1, plane), mismatch_row,
168 mismatch_col, mismatch_pix1, mismatch_pix2)) {
169 if (mismatch_plane != NULL) *mismatch_plane = plane;
170 return false;
171 }
Geza Loree0dcab92016-05-09 13:19:34 +0100172 }
173
174 return true;
Yaowu Xuc953aea2012-08-30 13:43:15 -0700175}
John Koleszarb9180fc2012-05-16 15:27:00 -0700176
Yaowu Xuf883b422016-08-30 14:01:10 -0700177void EncoderTest::MismatchHook(const aom_image_t *img_enc,
178 const aom_image_t *img_dec) {
Geza Lore7de2ba32016-06-19 21:35:01 +0100179 int mismatch_row = 0;
180 int mismatch_col = 0;
181 int mismatch_plane = 0;
182 int mismatch_pix_enc = 0;
183 int mismatch_pix_dec = 0;
Geza Loree0dcab92016-05-09 13:19:34 +0100184
clang-format3a826f12016-08-11 17:46:05 -0700185 ASSERT_FALSE(compare_img(img_enc, img_dec, &mismatch_row, &mismatch_col,
186 &mismatch_plane, &mismatch_pix_enc,
Geza Loree0dcab92016-05-09 13:19:34 +0100187 &mismatch_pix_dec));
188
clang-format3a826f12016-08-11 17:46:05 -0700189 GTEST_FAIL() << "Encode/Decode mismatch found:" << std::endl
190 << " pixel value enc/dec: " << mismatch_pix_enc << "/"
191 << mismatch_pix_dec << std::endl
192 << " plane: " << mismatch_plane << std::endl
193 << " row/col: " << mismatch_row << "/"
194 << mismatch_col << std::endl;
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800195}
196
John Koleszarb9180fc2012-05-16 15:27:00 -0700197void EncoderTest::RunLoop(VideoSource *video) {
Yaowu Xuf883b422016-08-30 14:01:10 -0700198 aom_codec_dec_cfg_t dec_cfg = aom_codec_dec_cfg_t();
Sebastien Alaiwan8b7a4e12017-06-13 11:25:57 +0200199 dec_cfg.allow_lowbitdepth = 1;
hkuang93536072014-11-20 15:39:56 -0800200
Adrian Grange30f58b52012-10-02 09:36:41 -0700201 stats_.Reset();
202
Ronald S. Bultjee189edf2013-03-01 12:43:10 -0800203 ASSERT_TRUE(passes_ == 1 || passes_ == 2);
John Koleszarb9180fc2012-05-16 15:27:00 -0700204 for (unsigned int pass = 0; pass < passes_; pass++) {
James Zern51ebb9a2012-08-08 14:16:08 -0700205 last_pts_ = 0;
206
John Koleszarb9180fc2012-05-16 15:27:00 -0700207 if (passes_ == 1)
Yaowu Xuf883b422016-08-30 14:01:10 -0700208 cfg_.g_pass = AOM_RC_ONE_PASS;
John Koleszarb9180fc2012-05-16 15:27:00 -0700209 else if (pass == 0)
Yaowu Xuf883b422016-08-30 14:01:10 -0700210 cfg_.g_pass = AOM_RC_FIRST_PASS;
John Koleszarb9180fc2012-05-16 15:27:00 -0700211 else
Yaowu Xuf883b422016-08-30 14:01:10 -0700212 cfg_.g_pass = AOM_RC_LAST_PASS;
John Koleszarb9180fc2012-05-16 15:27:00 -0700213
214 BeginPassHook(pass);
Alex Converse1c1bc942016-07-21 11:36:41 -0700215 testing::internal::scoped_ptr<Encoder> encoder(
Sean DuBois47cc2552018-01-23 07:44:16 +0000216 codec_->CreateEncoder(cfg_, init_flags_, &stats_));
Alex Converse1c1bc942016-07-21 11:36:41 -0700217 ASSERT_TRUE(encoder.get() != NULL);
hkuang93536072014-11-20 15:39:56 -0800218
Alex Converse1c1bc942016-07-21 11:36:41 -0700219 ASSERT_NO_FATAL_FAILURE(video->Begin());
Yunqing Wang36664782014-12-12 14:34:30 -0800220 encoder->InitEncoder(video);
Thomas Daede80826142017-03-20 15:44:24 -0700221
222 if (mode_ == kRealTime) {
223 encoder->Control(AOME_SET_ENABLEAUTOALTREF, 0);
224 }
225
James Zernd1ff1e92015-08-27 16:05:52 -0700226 ASSERT_FALSE(::testing::Test::HasFatalFailure());
Yunqing Wang36664782014-12-12 14:34:30 -0800227
hkuang93536072014-11-20 15:39:56 -0800228 unsigned long dec_init_flags = 0; // NOLINT
229 // Use fragment decoder if encoder outputs partitions.
230 // NOTE: fragment decoder and partition encoder are only supported by VP8.
Yaowu Xuf883b422016-08-30 14:01:10 -0700231 if (init_flags_ & AOM_CODEC_USE_OUTPUT_PARTITION)
232 dec_init_flags |= AOM_CODEC_USE_INPUT_FRAGMENTS;
Alex Converse1c1bc942016-07-21 11:36:41 -0700233 testing::internal::scoped_ptr<Decoder> decoder(
James Zern3b96b762017-03-24 17:12:19 -0700234 codec_->CreateDecoder(dec_cfg, dec_init_flags));
Yaowu Xuf883b422016-08-30 14:01:10 -0700235#if CONFIG_AV1 && CONFIG_EXT_TILE
236 if (decoder->IsAV1()) {
Yunqing Wang8e5e3382016-05-05 16:42:57 -0700237 // Set dec_cfg.tile_row = -1 and dec_cfg.tile_col = -1 so that the whole
238 // frame is decoded.
Yunqing Wang8ae64a92018-01-12 12:26:44 -0800239 decoder->Control(AV1_SET_TILE_MODE, cfg_.large_scale_tile);
Yaowu Xuf883b422016-08-30 14:01:10 -0700240 decoder->Control(AV1_SET_DECODE_TILE_ROW, -1);
241 decoder->Control(AV1_SET_DECODE_TILE_COL, -1);
Yunqing Wang8e5e3382016-05-05 16:42:57 -0700242 }
243#endif
244
John Koleszarb9180fc2012-05-16 15:27:00 -0700245 bool again;
Yunqing Wang36664782014-12-12 14:34:30 -0800246 for (again = true; again; video->Next()) {
Deb Mukherjee0d8723f2013-08-19 14:16:26 -0700247 again = (video->img() != NULL);
John Koleszarb9180fc2012-05-16 15:27:00 -0700248
249 PreEncodeFrameHook(video);
Alex Converse1c1bc942016-07-21 11:36:41 -0700250 PreEncodeFrameHook(video, encoder.get());
John Koleszar706cafe2013-01-18 11:51:12 -0800251 encoder->EncodeFrame(video, frame_flags_);
John Koleszarb9180fc2012-05-16 15:27:00 -0700252
John Koleszar706cafe2013-01-18 11:51:12 -0800253 CxDataIterator iter = encoder->GetCxData();
John Koleszarb9180fc2012-05-16 15:27:00 -0700254
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800255 bool has_cxdata = false;
256 bool has_dxdata = false;
Yaowu Xuf883b422016-08-30 14:01:10 -0700257 while (const aom_codec_cx_pkt_t *pkt = iter.Next()) {
John Koleszar522d4bf2013-03-05 12:23:34 -0800258 pkt = MutateEncoderOutputHook(pkt);
John Koleszarb9180fc2012-05-16 15:27:00 -0700259 again = true;
Adrian Grangee6109db2012-10-02 11:27:29 -0700260 switch (pkt->kind) {
Yaowu Xuf883b422016-08-30 14:01:10 -0700261 case AOM_CODEC_CX_FRAME_PKT:
Adrian Grangee6109db2012-10-02 11:27:29 -0700262 has_cxdata = true;
Alex Converse1c1bc942016-07-21 11:36:41 -0700263 if (decoder.get() != NULL && DoDecode()) {
Yaowu Xuf883b422016-08-30 14:01:10 -0700264 aom_codec_err_t res_dec = decoder->DecodeFrame(
clang-format3a826f12016-08-11 17:46:05 -0700265 (const uint8_t *)pkt->data.frame.buf, pkt->data.frame.sz);
Jim Bankoski943e4322014-07-17 06:31:50 -0700266
Urvang Joshid71a2312016-07-14 12:33:48 -0700267 if (!HandleDecodeResult(res_dec, decoder.get())) break;
Jim Bankoski943e4322014-07-17 06:31:50 -0700268
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800269 has_dxdata = true;
270 }
Adrian Grangee6109db2012-10-02 11:27:29 -0700271 ASSERT_GE(pkt->data.frame.pts, last_pts_);
272 last_pts_ = pkt->data.frame.pts;
273 FramePktHook(pkt);
274 break;
275
Yaowu Xuf883b422016-08-30 14:01:10 -0700276 case AOM_CODEC_PSNR_PKT: PSNRPktHook(pkt); break;
Adrian Grangee6109db2012-10-02 11:27:29 -0700277
clang-format3a826f12016-08-11 17:46:05 -0700278 default: break;
Adrian Grangee6109db2012-10-02 11:27:29 -0700279 }
John Koleszarb9180fc2012-05-16 15:27:00 -0700280 }
281
hkuang93536072014-11-20 15:39:56 -0800282 // Flush the decoder when there are no more fragments.
Yaowu Xuf883b422016-08-30 14:01:10 -0700283 if ((init_flags_ & AOM_CODEC_USE_OUTPUT_PARTITION) && has_dxdata) {
284 const aom_codec_err_t res_dec = decoder->DecodeFrame(NULL, 0);
Urvang Joshid71a2312016-07-14 12:33:48 -0700285 if (!HandleDecodeResult(res_dec, decoder.get())) break;
hkuang93536072014-11-20 15:39:56 -0800286 }
287
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800288 if (has_dxdata && has_cxdata) {
Yaowu Xuf883b422016-08-30 14:01:10 -0700289 const aom_image_t *img_enc = encoder->GetPreviewFrame();
John Koleszar706cafe2013-01-18 11:51:12 -0800290 DxDataIterator dec_iter = decoder->GetDxData();
Yaowu Xuf883b422016-08-30 14:01:10 -0700291 const aom_image_t *img_dec = dec_iter.Next();
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800292 if (img_enc && img_dec) {
clang-format3a826f12016-08-11 17:46:05 -0700293 const bool res =
294 compare_img(img_enc, img_dec, NULL, NULL, NULL, NULL, NULL);
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800295 if (!res) { // Mismatch
296 MismatchHook(img_enc, img_dec);
297 }
Yaowu Xuc953aea2012-08-30 13:43:15 -0700298 }
clang-format3a826f12016-08-11 17:46:05 -0700299 if (img_dec) DecompressedFrameHook(*img_dec, video->pts());
Yaowu Xuc953aea2012-08-30 13:43:15 -0700300 }
clang-format3a826f12016-08-11 17:46:05 -0700301 if (!Continue()) break;
John Koleszarb9180fc2012-05-16 15:27:00 -0700302 }
303
304 EndPassHook();
305
clang-format3a826f12016-08-11 17:46:05 -0700306 if (!Continue()) break;
John Koleszarb9180fc2012-05-16 15:27:00 -0700307 }
308}
Deb Mukherjee01cafaa2013-01-15 06:43:35 -0800309
Yaowu Xuc27fc142016-08-22 16:08:15 -0700310} // namespace libaom_test