| /* | 
 |  * Copyright (c) 2021, Alliance for Open Media. All rights reserved | 
 |  * | 
 |  * This source code is subject to the terms of the BSD 3-Clause Clear License | 
 |  * and the Alliance for Open Media Patent License 1.0. If the BSD 3-Clause Clear | 
 |  * License was not distributed with this source code in the LICENSE file, you | 
 |  * can obtain it at aomedia.org/license/software-license/bsd-3-c-c/.  If the | 
 |  * Alliance for Open Media Patent License 1.0 was not distributed with this | 
 |  * source code in the PATENTS file, you can obtain it at | 
 |  * aomedia.org/license/patent-license/. | 
 |  */ | 
 |  | 
 | #include <string> | 
 | #include <tuple> | 
 |  | 
 | #include "config/aom_version.h" | 
 |  | 
 | #include "aom_ports/aom_timer.h" | 
 | #include "common/ivfenc.h" | 
 | #include "test/codec_factory.h" | 
 | #include "test/decode_test_driver.h" | 
 | #include "test/encode_test_driver.h" | 
 | #include "test/i420_video_source.h" | 
 | #include "test/ivf_video_source.h" | 
 | #include "test/md5_helper.h" | 
 | #include "test/util.h" | 
 | #include "test/webm_video_source.h" | 
 |  | 
 | using std::make_tuple; | 
 |  | 
 | namespace { | 
 |  | 
 | #define VIDEO_NAME 0 | 
 | #define THREADS 1 | 
 |  | 
 | const double kUsecsInSec = 1000000.0; | 
 | const char kNewEncodeOutputFile[] = "new_encode.ivf"; | 
 |  | 
 | /* | 
 |  DecodePerfTest takes a tuple of filename + number of threads to decode with | 
 |  */ | 
 | typedef std::tuple<const char *, unsigned> DecodePerfParam; | 
 |  | 
 | // TODO(jimbankoski): Add actual test vectors here when available. | 
 | // const DecodePerfParam kAV1DecodePerfVectors[] = {}; | 
 |  | 
 | /* | 
 |  In order to reflect real world performance as much as possible, Perf tests | 
 |  *DO NOT* do any correctness checks. Please run them alongside correctness | 
 |  tests to ensure proper codec integrity. Furthermore, in this test we | 
 |  deliberately limit the amount of system calls we make to avoid OS | 
 |  preemption. | 
 |  | 
 |  TODO(joshualitt) create a more detailed perf measurement test to collect | 
 |    power/temp/min max frame decode times/etc | 
 |  */ | 
 |  | 
 | class DecodePerfTest : public ::testing::TestWithParam<DecodePerfParam> {}; | 
 |  | 
 | TEST_P(DecodePerfTest, PerfTest) { | 
 |   const char *const video_name = GET_PARAM(VIDEO_NAME); | 
 |   const unsigned threads = GET_PARAM(THREADS); | 
 |  | 
 |   libaom_test::WebMVideoSource video(video_name); | 
 |   video.Init(); | 
 |  | 
 |   aom_codec_dec_cfg_t cfg = aom_codec_dec_cfg_t(); | 
 |   cfg.threads = threads; | 
 |   cfg.allow_lowbitdepth = 1; | 
 |   libaom_test::AV1Decoder decoder(cfg, 0); | 
 |  | 
 |   aom_usec_timer t; | 
 |   aom_usec_timer_start(&t); | 
 |  | 
 |   for (video.Begin(); video.cxdata() != NULL; video.Next()) { | 
 |     decoder.DecodeFrame(video.cxdata(), video.frame_size()); | 
 |   } | 
 |  | 
 |   aom_usec_timer_mark(&t); | 
 |   const double elapsed_secs = double(aom_usec_timer_elapsed(&t)) / kUsecsInSec; | 
 |   const unsigned frames = video.frame_number(); | 
 |   const double fps = double(frames) / elapsed_secs; | 
 |  | 
 |   printf("{\n"); | 
 |   printf("\t\"type\" : \"decode_perf_test\",\n"); | 
 |   printf("\t\"version\" : \"%s\",\n", VERSION_STRING_NOSP); | 
 |   printf("\t\"videoName\" : \"%s\",\n", video_name); | 
 |   printf("\t\"threadCount\" : %u,\n", threads); | 
 |   printf("\t\"decodeTimeSecs\" : %f,\n", elapsed_secs); | 
 |   printf("\t\"totalFrames\" : %u,\n", frames); | 
 |   printf("\t\"framesPerSecond\" : %f\n", fps); | 
 |   printf("}\n"); | 
 | } | 
 |  | 
 | // TODO(jimbankoski): Enabled when we have actual AV1 Decode vectors. | 
 | // INSTANTIATE_TEST_SUITE_P(AV1, DecodePerfTest, | 
 | //                        ::testing::ValuesIn(kAV1DecodePerfVectors)); | 
 |  | 
 | class AV1NewEncodeDecodePerfTest | 
 |     : public ::libaom_test::CodecTestWithParam<libaom_test::TestMode>, | 
 |       public ::libaom_test::EncoderTest { | 
 |  protected: | 
 |   AV1NewEncodeDecodePerfTest() | 
 |       : EncoderTest(GET_PARAM(0)), encoding_mode_(GET_PARAM(1)), speed_(0), | 
 |         outfile_(0), out_frames_(0) {} | 
 |  | 
 |   virtual ~AV1NewEncodeDecodePerfTest() {} | 
 |  | 
 |   virtual void SetUp() { | 
 |     InitializeConfig(); | 
 |     SetMode(encoding_mode_); | 
 |  | 
 |     cfg_.g_lag_in_frames = 25; | 
 |     cfg_.rc_min_quantizer = 8; | 
 |     cfg_.rc_max_quantizer = 224; | 
 |     cfg_.rc_dropframe_thresh = 0; | 
 |     cfg_.rc_undershoot_pct = 50; | 
 |     cfg_.rc_overshoot_pct = 50; | 
 |     cfg_.rc_buf_sz = 1000; | 
 |     cfg_.rc_buf_initial_sz = 500; | 
 |     cfg_.rc_buf_optimal_sz = 600; | 
 |     cfg_.rc_end_usage = AOM_VBR; | 
 |   } | 
 |  | 
 |   virtual void PreEncodeFrameHook(::libaom_test::VideoSource *video, | 
 |                                   ::libaom_test::Encoder *encoder) { | 
 |     if (video->frame() == 0) { | 
 |       encoder->Control(AOME_SET_CPUUSED, speed_); | 
 |       encoder->Control(AV1E_SET_FRAME_PARALLEL_DECODING, 1); | 
 |       encoder->Control(AV1E_SET_TILE_COLUMNS, 2); | 
 |     } | 
 |   } | 
 |  | 
 |   virtual void BeginPassHook(unsigned int /*pass*/) { | 
 |     const char *const env = getenv("LIBAOM_TEST_DATA_PATH"); | 
 |     const std::string data_path(env ? env : "."); | 
 |     const std::string path_to_source = data_path + "/" + kNewEncodeOutputFile; | 
 |     outfile_ = fopen(path_to_source.c_str(), "wb"); | 
 |     ASSERT_TRUE(outfile_ != NULL); | 
 |   } | 
 |  | 
 |   virtual void EndPassHook() { | 
 |     if (outfile_ != NULL) { | 
 |       if (!fseek(outfile_, 0, SEEK_SET)) | 
 |         ivf_write_file_header(outfile_, &cfg_, AV1_FOURCC, out_frames_); | 
 |       fclose(outfile_); | 
 |       outfile_ = NULL; | 
 |     } | 
 |   } | 
 |  | 
 |   virtual void FramePktHook(const aom_codec_cx_pkt_t *pkt) { | 
 |     ++out_frames_; | 
 |  | 
 |     // Write initial file header if first frame. | 
 |     if (pkt->data.frame.pts == 0) | 
 |       ivf_write_file_header(outfile_, &cfg_, AV1_FOURCC, out_frames_); | 
 |  | 
 |     // Write frame header and data. | 
 |     ivf_write_frame_header(outfile_, out_frames_, pkt->data.frame.sz); | 
 |     ASSERT_EQ(fwrite(pkt->data.frame.buf, 1, pkt->data.frame.sz, outfile_), | 
 |               pkt->data.frame.sz); | 
 |   } | 
 |  | 
 |   virtual bool DoDecode() const { return false; } | 
 |  | 
 |   void set_speed(unsigned int speed) { speed_ = speed; } | 
 |  | 
 |  private: | 
 |   libaom_test::TestMode encoding_mode_; | 
 |   uint32_t speed_; | 
 |   FILE *outfile_; | 
 |   uint32_t out_frames_; | 
 | }; | 
 |  | 
 | struct EncodePerfTestVideo { | 
 |   EncodePerfTestVideo(const char *name_, uint32_t width_, uint32_t height_, | 
 |                       uint32_t bitrate_, int frames_) | 
 |       : name(name_), width(width_), height(height_), bitrate(bitrate_), | 
 |         frames(frames_) {} | 
 |   const char *name; | 
 |   uint32_t width; | 
 |   uint32_t height; | 
 |   uint32_t bitrate; | 
 |   int frames; | 
 | }; | 
 |  | 
 | const EncodePerfTestVideo kAV1EncodePerfTestVectors[] = { | 
 |   EncodePerfTestVideo("niklas_1280_720_30.yuv", 1280, 720, 600, 470), | 
 | }; | 
 |  | 
 | TEST_P(AV1NewEncodeDecodePerfTest, PerfTest) { | 
 |   SetUp(); | 
 |  | 
 |   // TODO(JBB): Make this work by going through the set of given files. | 
 |   const int i = 0; | 
 |   const aom_rational timebase = { 33333333, 1000000000 }; | 
 |   cfg_.g_timebase = timebase; | 
 |   cfg_.rc_target_bitrate = kAV1EncodePerfTestVectors[i].bitrate; | 
 |  | 
 |   init_flags_ = AOM_CODEC_USE_PSNR; | 
 |  | 
 |   const char *video_name = kAV1EncodePerfTestVectors[i].name; | 
 |   libaom_test::I420VideoSource video( | 
 |       video_name, kAV1EncodePerfTestVectors[i].width, | 
 |       kAV1EncodePerfTestVectors[i].height, timebase.den, timebase.num, 0, | 
 |       kAV1EncodePerfTestVectors[i].frames); | 
 |   set_speed(2); | 
 |  | 
 |   ASSERT_NO_FATAL_FAILURE(RunLoop(&video)); | 
 |  | 
 |   const uint32_t threads = 4; | 
 |  | 
 |   libaom_test::IVFVideoSource decode_video(kNewEncodeOutputFile); | 
 |   decode_video.Init(); | 
 |  | 
 |   aom_codec_dec_cfg_t cfg = aom_codec_dec_cfg_t(); | 
 |   cfg.threads = threads; | 
 |   cfg.allow_lowbitdepth = 1; | 
 |   libaom_test::AV1Decoder decoder(cfg, 0); | 
 |  | 
 |   aom_usec_timer t; | 
 |   aom_usec_timer_start(&t); | 
 |  | 
 |   for (decode_video.Begin(); decode_video.cxdata() != NULL; | 
 |        decode_video.Next()) { | 
 |     decoder.DecodeFrame(decode_video.cxdata(), decode_video.frame_size()); | 
 |   } | 
 |  | 
 |   aom_usec_timer_mark(&t); | 
 |   const double elapsed_secs = | 
 |       static_cast<double>(aom_usec_timer_elapsed(&t)) / kUsecsInSec; | 
 |   const unsigned decode_frames = decode_video.frame_number(); | 
 |   const double fps = static_cast<double>(decode_frames) / elapsed_secs; | 
 |  | 
 |   printf("{\n"); | 
 |   printf("\t\"type\" : \"decode_perf_test\",\n"); | 
 |   printf("\t\"version\" : \"%s\",\n", VERSION_STRING_NOSP); | 
 |   printf("\t\"videoName\" : \"%s\",\n", kNewEncodeOutputFile); | 
 |   printf("\t\"threadCount\" : %u,\n", threads); | 
 |   printf("\t\"decodeTimeSecs\" : %f,\n", elapsed_secs); | 
 |   printf("\t\"totalFrames\" : %u,\n", decode_frames); | 
 |   printf("\t\"framesPerSecond\" : %f\n", fps); | 
 |   printf("}\n"); | 
 | } | 
 |  | 
 | AV1_INSTANTIATE_TEST_SUITE(AV1NewEncodeDecodePerfTest, | 
 |                            ::testing::Values(::libaom_test::kTwoPassGood)); | 
 | }  // namespace |