| Index: content/common/gpu/media/video_encode_accelerator_unittest.cc
|
| diff --git a/content/common/gpu/media/video_encode_accelerator_unittest.cc b/content/common/gpu/media/video_encode_accelerator_unittest.cc
|
| deleted file mode 100644
|
| index 3f3f9ef66d0a709804ed0c7a54fb885cf1dccd96..0000000000000000000000000000000000000000
|
| --- a/content/common/gpu/media/video_encode_accelerator_unittest.cc
|
| +++ /dev/null
|
| @@ -1,1829 +0,0 @@
|
| -// Copyright 2013 The Chromium Authors. All rights reserved.
|
| -// Use of this source code is governed by a BSD-style license that can be
|
| -// found in the LICENSE file.
|
| -
|
| -#include <inttypes.h>
|
| -#include <stddef.h>
|
| -#include <stdint.h>
|
| -
|
| -#include <algorithm>
|
| -#include <memory>
|
| -#include <queue>
|
| -#include <string>
|
| -#include <utility>
|
| -
|
| -#include "base/at_exit.h"
|
| -#include "base/bind.h"
|
| -#include "base/command_line.h"
|
| -#include "base/files/file_util.h"
|
| -#include "base/files/memory_mapped_file.h"
|
| -#include "base/macros.h"
|
| -#include "base/memory/scoped_vector.h"
|
| -#include "base/message_loop/message_loop.h"
|
| -#include "base/numerics/safe_conversions.h"
|
| -#include "base/process/process_handle.h"
|
| -#include "base/strings/string_number_conversions.h"
|
| -#include "base/strings/string_split.h"
|
| -#include "base/strings/stringprintf.h"
|
| -#include "base/threading/thread.h"
|
| -#include "base/threading/thread_checker.h"
|
| -#include "base/time/time.h"
|
| -#include "base/timer/timer.h"
|
| -#include "build/build_config.h"
|
| -#include "content/common/gpu/media/video_accelerator_unittest_helpers.h"
|
| -#include "media/base/bind_to_current_loop.h"
|
| -#include "media/base/bitstream_buffer.h"
|
| -#include "media/base/cdm_context.h"
|
| -#include "media/base/decoder_buffer.h"
|
| -#include "media/base/media_util.h"
|
| -#include "media/base/test_data_util.h"
|
| -#include "media/base/video_decoder.h"
|
| -#include "media/base/video_frame.h"
|
| -#include "media/filters/ffmpeg_glue.h"
|
| -#include "media/filters/ffmpeg_video_decoder.h"
|
| -#include "media/filters/h264_parser.h"
|
| -#include "media/filters/ivf_parser.h"
|
| -#include "media/video/fake_video_encode_accelerator.h"
|
| -#include "media/video/video_encode_accelerator.h"
|
| -#include "testing/gtest/include/gtest/gtest.h"
|
| -
|
| -#if defined(OS_CHROMEOS)
|
| -#if defined(ARCH_CPU_ARMEL) || (defined(USE_OZONE) && defined(USE_V4L2_CODEC))
|
| -#include "content/common/gpu/media/v4l2_video_encode_accelerator.h"
|
| -#endif
|
| -#if defined(ARCH_CPU_X86_FAMILY)
|
| -#include "content/common/gpu/media/vaapi_video_encode_accelerator.h"
|
| -#include "content/common/gpu/media/vaapi_wrapper.h"
|
| -// Status has been defined as int in Xlib.h.
|
| -#undef Status
|
| -#endif // defined(ARCH_CPU_X86_FAMILY)
|
| -#elif defined(OS_MACOSX)
|
| -#include "content/common/gpu/media/vt_video_encode_accelerator_mac.h"
|
| -#else
|
| -#error The VideoEncodeAcceleratorUnittest is not supported on this platform.
|
| -#endif
|
| -
|
| -using media::VideoEncodeAccelerator;
|
| -
|
| -namespace content {
|
| -namespace {
|
| -
|
| -const media::VideoPixelFormat kInputFormat = media::PIXEL_FORMAT_I420;
|
| -
|
| -// The absolute differences between original frame and decoded frame usually
|
| -// ranges aroud 1 ~ 7. So we pick 10 as an extreme value to detect abnormal
|
| -// decoded frames.
|
| -const double kDecodeSimilarityThreshold = 10.0;
|
| -
|
| -// Arbitrarily chosen to add some depth to the pipeline.
|
| -const unsigned int kNumOutputBuffers = 4;
|
| -const unsigned int kNumExtraInputFrames = 4;
|
| -// Maximum delay between requesting a keyframe and receiving one, in frames.
|
| -// Arbitrarily chosen as a reasonable requirement.
|
| -const unsigned int kMaxKeyframeDelay = 4;
|
| -// Default initial bitrate.
|
| -const uint32_t kDefaultBitrate = 2000000;
|
| -// Default ratio of requested_subsequent_bitrate to initial_bitrate
|
| -// (see test parameters below) if one is not provided.
|
| -const double kDefaultSubsequentBitrateRatio = 2.0;
|
| -// Default initial framerate.
|
| -const uint32_t kDefaultFramerate = 30;
|
| -// Default ratio of requested_subsequent_framerate to initial_framerate
|
| -// (see test parameters below) if one is not provided.
|
| -const double kDefaultSubsequentFramerateRatio = 0.1;
|
| -// Tolerance factor for how encoded bitrate can differ from requested bitrate.
|
| -const double kBitrateTolerance = 0.1;
|
| -// Minimum required FPS throughput for the basic performance test.
|
| -const uint32_t kMinPerfFPS = 30;
|
| -// Minimum (arbitrary) number of frames required to enforce bitrate requirements
|
| -// over. Streams shorter than this may be too short to realistically require
|
| -// an encoder to be able to converge to the requested bitrate over.
|
| -// The input stream will be looped as many times as needed in bitrate tests
|
| -// to reach at least this number of frames before calculating final bitrate.
|
| -const unsigned int kMinFramesForBitrateTests = 300;
|
| -// The percentiles to measure for encode latency.
|
| -const unsigned int kLoggedLatencyPercentiles[] = {50, 75, 95};
|
| -
|
| -// The syntax of multiple test streams is:
|
| -// test-stream1;test-stream2;test-stream3
|
| -// The syntax of each test stream is:
|
| -// "in_filename:width:height:profile:out_filename:requested_bitrate
|
| -// :requested_framerate:requested_subsequent_bitrate
|
| -// :requested_subsequent_framerate"
|
| -// - |in_filename| must be an I420 (YUV planar) raw stream
|
| -// (see http://www.fourcc.org/yuv.php#IYUV).
|
| -// - |width| and |height| are in pixels.
|
| -// - |profile| to encode into (values of media::VideoCodecProfile).
|
| -// - |out_filename| filename to save the encoded stream to (optional). The
|
| -// format for H264 is Annex-B byte stream. The format for VP8 is IVF. Output
|
| -// stream is saved for the simple encode test only. H264 raw stream and IVF
|
| -// can be used as input of VDA unittest. H264 raw stream can be played by
|
| -// "mplayer -fps 25 out.h264" and IVF can be played by mplayer directly.
|
| -// Helpful description: http://wiki.multimedia.cx/index.php?title=IVF
|
| -// Further parameters are optional (need to provide preceding positional
|
| -// parameters if a specific subsequent parameter is required):
|
| -// - |requested_bitrate| requested bitrate in bits per second.
|
| -// - |requested_framerate| requested initial framerate.
|
| -// - |requested_subsequent_bitrate| bitrate to switch to in the middle of the
|
| -// stream.
|
| -// - |requested_subsequent_framerate| framerate to switch to in the middle
|
| -// of the stream.
|
| -// Bitrate is only forced for tests that test bitrate.
|
| -const char* g_default_in_filename = "bear_320x192_40frames.yuv";
|
| -#if !defined(OS_MACOSX)
|
| -const char* g_default_in_parameters = ":320:192:1:out.h264:200000";
|
| -#else
|
| -const char* g_default_in_parameters = ":320:192:0:out.h264:200000";
|
| -#endif
|
| -
|
| -// Enabled by including a --fake_encoder flag to the command line invoking the
|
| -// test.
|
| -bool g_fake_encoder = false;
|
| -
|
| -// Environment to store test stream data for all test cases.
|
| -class VideoEncodeAcceleratorTestEnvironment;
|
| -VideoEncodeAcceleratorTestEnvironment* g_env;
|
| -
|
| -// The number of frames to be encoded. This variable is set by the switch
|
| -// "--num_frames_to_encode". Ignored if 0.
|
| -int g_num_frames_to_encode = 0;
|
| -
|
| -struct TestStream {
|
| - TestStream()
|
| - : num_frames(0),
|
| - aligned_buffer_size(0),
|
| - requested_bitrate(0),
|
| - requested_framerate(0),
|
| - requested_subsequent_bitrate(0),
|
| - requested_subsequent_framerate(0) {}
|
| - ~TestStream() {}
|
| -
|
| - gfx::Size visible_size;
|
| - gfx::Size coded_size;
|
| - unsigned int num_frames;
|
| -
|
| - // Original unaligned input file name provided as an argument to the test.
|
| - // And the file must be an I420 (YUV planar) raw stream.
|
| - std::string in_filename;
|
| -
|
| - // A temporary file used to prepare aligned input buffers of |in_filename|.
|
| - // The file makes sure starting address of YUV planes are 64 byte-aligned.
|
| - base::FilePath aligned_in_file;
|
| -
|
| - // The memory mapping of |aligned_in_file|
|
| - base::MemoryMappedFile mapped_aligned_in_file;
|
| -
|
| - // Byte size of a frame of |aligned_in_file|.
|
| - size_t aligned_buffer_size;
|
| -
|
| - // Byte size for each aligned plane of a frame
|
| - std::vector<size_t> aligned_plane_size;
|
| -
|
| - std::string out_filename;
|
| - media::VideoCodecProfile requested_profile;
|
| - unsigned int requested_bitrate;
|
| - unsigned int requested_framerate;
|
| - unsigned int requested_subsequent_bitrate;
|
| - unsigned int requested_subsequent_framerate;
|
| -};
|
| -
|
| -inline static size_t Align64Bytes(size_t value) {
|
| - return (value + 63) & ~63;
|
| -}
|
| -
|
| -// Write |data| of |size| bytes at |offset| bytes into |file|.
|
| -static bool WriteFile(base::File* file,
|
| - const off_t offset,
|
| - const uint8_t* data,
|
| - size_t size) {
|
| - size_t written_bytes = 0;
|
| - while (written_bytes < size) {
|
| - int bytes = file->Write(offset + written_bytes,
|
| - reinterpret_cast<const char*>(data + written_bytes),
|
| - size - written_bytes);
|
| - if (bytes <= 0)
|
| - return false;
|
| - written_bytes += bytes;
|
| - }
|
| - return true;
|
| -}
|
| -
|
| -// Return the |percentile| from a sorted vector.
|
| -static base::TimeDelta Percentile(
|
| - const std::vector<base::TimeDelta>& sorted_values,
|
| - unsigned int percentile) {
|
| - size_t size = sorted_values.size();
|
| - LOG_ASSERT(size > 0UL);
|
| - LOG_ASSERT(percentile <= 100UL);
|
| - // Use Nearest Rank method in http://en.wikipedia.org/wiki/Percentile.
|
| - int index =
|
| - std::max(static_cast<int>(ceil(0.01f * percentile * size)) - 1, 0);
|
| - return sorted_values[index];
|
| -}
|
| -
|
| -static bool IsH264(media::VideoCodecProfile profile) {
|
| - return profile >= media::H264PROFILE_MIN && profile <= media::H264PROFILE_MAX;
|
| -}
|
| -
|
| -static bool IsVP8(media::VideoCodecProfile profile) {
|
| - return profile >= media::VP8PROFILE_MIN && profile <= media::VP8PROFILE_MAX;
|
| -}
|
| -
|
| -// ARM performs CPU cache management with CPU cache line granularity. We thus
|
| -// need to ensure our buffers are CPU cache line-aligned (64 byte-aligned).
|
| -// Otherwise newer kernels will refuse to accept them, and on older kernels
|
| -// we'll be treating ourselves to random corruption.
|
| -// Since we are just mapping and passing chunks of the input file directly to
|
| -// the VEA as input frames to avoid copying large chunks of raw data on each
|
| -// frame and thus affecting performance measurements, we have to prepare a
|
| -// temporary file with all planes aligned to 64-byte boundaries beforehand.
|
| -static void CreateAlignedInputStreamFile(const gfx::Size& coded_size,
|
| - TestStream* test_stream) {
|
| - // Test case may have many encoders and memory should be prepared once.
|
| - if (test_stream->coded_size == coded_size &&
|
| - test_stream->mapped_aligned_in_file.IsValid())
|
| - return;
|
| -
|
| - // All encoders in multiple encoder test reuse the same test_stream, make
|
| - // sure they requested the same coded_size
|
| - ASSERT_TRUE(!test_stream->mapped_aligned_in_file.IsValid() ||
|
| - coded_size == test_stream->coded_size);
|
| - test_stream->coded_size = coded_size;
|
| -
|
| - size_t num_planes = media::VideoFrame::NumPlanes(kInputFormat);
|
| - std::vector<std::vector<uint8_t>> padding(num_planes);
|
| - std::vector<size_t> coded_bpl(num_planes);
|
| - std::vector<size_t> visible_bpl(num_planes);
|
| - std::vector<size_t> visible_plane_rows(num_planes);
|
| -
|
| - // Calculate padding in bytes to be added after each plane required to keep
|
| - // starting addresses of all planes at a 64 byte boudnary. This padding will
|
| - // be added after each plane when copying to the temporary file.
|
| - // At the same time we also need to take into account coded_size requested by
|
| - // the VEA; each row of visible_bpl bytes in the original file needs to be
|
| - // copied into a row of coded_bpl bytes in the aligned file.
|
| - for (size_t i = 0; i < num_planes; i++) {
|
| - const size_t size =
|
| - media::VideoFrame::PlaneSize(kInputFormat, i, coded_size).GetArea();
|
| - test_stream->aligned_plane_size.push_back(Align64Bytes(size));
|
| - test_stream->aligned_buffer_size += test_stream->aligned_plane_size.back();
|
| -
|
| - coded_bpl[i] =
|
| - media::VideoFrame::RowBytes(i, kInputFormat, coded_size.width());
|
| - visible_bpl[i] = media::VideoFrame::RowBytes(
|
| - i, kInputFormat, test_stream->visible_size.width());
|
| - visible_plane_rows[i] = media::VideoFrame::Rows(
|
| - i, kInputFormat, test_stream->visible_size.height());
|
| - const size_t padding_rows =
|
| - media::VideoFrame::Rows(i, kInputFormat, coded_size.height()) -
|
| - visible_plane_rows[i];
|
| - padding[i].resize(padding_rows * coded_bpl[i] + Align64Bytes(size) - size);
|
| - }
|
| -
|
| - base::MemoryMappedFile src_file;
|
| - LOG_ASSERT(src_file.Initialize(base::FilePath(test_stream->in_filename)));
|
| - LOG_ASSERT(base::CreateTemporaryFile(&test_stream->aligned_in_file));
|
| -
|
| - size_t visible_buffer_size = media::VideoFrame::AllocationSize(
|
| - kInputFormat, test_stream->visible_size);
|
| - LOG_ASSERT(src_file.length() % visible_buffer_size == 0U)
|
| - << "Stream byte size is not a product of calculated frame byte size";
|
| -
|
| - test_stream->num_frames = src_file.length() / visible_buffer_size;
|
| - uint32_t flags = base::File::FLAG_CREATE_ALWAYS | base::File::FLAG_WRITE |
|
| - base::File::FLAG_READ;
|
| -
|
| - // Create a temporary file with coded_size length.
|
| - base::File dest_file(test_stream->aligned_in_file, flags);
|
| - LOG_ASSERT(test_stream->aligned_buffer_size > 0UL);
|
| - dest_file.SetLength(test_stream->aligned_buffer_size *
|
| - test_stream->num_frames);
|
| -
|
| - const uint8_t* src = src_file.data();
|
| - off_t dest_offset = 0;
|
| - for (size_t frame = 0; frame < test_stream->num_frames; frame++) {
|
| - for (size_t i = 0; i < num_planes; i++) {
|
| - // Assert that each plane of frame starts at 64 byte boundary.
|
| - ASSERT_EQ(dest_offset & 63, 0)
|
| - << "Planes of frame should be mapped at a 64 byte boundary";
|
| - for (size_t j = 0; j < visible_plane_rows[i]; j++) {
|
| - LOG_ASSERT(WriteFile(&dest_file, dest_offset, src, visible_bpl[i]));
|
| - src += visible_bpl[i];
|
| - dest_offset += coded_bpl[i];
|
| - }
|
| - if (!padding[i].empty()) {
|
| - LOG_ASSERT(WriteFile(&dest_file, dest_offset, &padding[i][0],
|
| - padding[i].size()));
|
| - dest_offset += padding[i].size();
|
| - }
|
| - }
|
| - }
|
| - LOG_ASSERT(
|
| - test_stream->mapped_aligned_in_file.Initialize(std::move(dest_file)));
|
| - // Assert that memory mapped of file starts at 64 byte boundary. So each
|
| - // plane of frames also start at 64 byte boundary.
|
| -
|
| - ASSERT_EQ(
|
| - reinterpret_cast<off_t>(test_stream->mapped_aligned_in_file.data()) & 63,
|
| - 0)
|
| - << "File should be mapped at a 64 byte boundary";
|
| -
|
| - LOG_ASSERT(test_stream->mapped_aligned_in_file.length() %
|
| - test_stream->aligned_buffer_size == 0U)
|
| - << "Stream byte size is not a product of calculated frame byte size";
|
| - LOG_ASSERT(test_stream->num_frames > 0UL);
|
| -}
|
| -
|
| -// Parse |data| into its constituent parts, set the various output fields
|
| -// accordingly, read in video stream, and store them to |test_streams|.
|
| -static void ParseAndReadTestStreamData(const base::FilePath::StringType& data,
|
| - ScopedVector<TestStream>* test_streams) {
|
| - // Split the string to individual test stream data.
|
| - std::vector<base::FilePath::StringType> test_streams_data = base::SplitString(
|
| - data, base::FilePath::StringType(1, ';'),
|
| - base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL);
|
| - LOG_ASSERT(test_streams_data.size() >= 1U) << data;
|
| -
|
| - // Parse each test stream data and read the input file.
|
| - for (size_t index = 0; index < test_streams_data.size(); ++index) {
|
| - std::vector<base::FilePath::StringType> fields = base::SplitString(
|
| - test_streams_data[index], base::FilePath::StringType(1, ':'),
|
| - base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL);
|
| - LOG_ASSERT(fields.size() >= 4U) << data;
|
| - LOG_ASSERT(fields.size() <= 9U) << data;
|
| - TestStream* test_stream = new TestStream();
|
| -
|
| - test_stream->in_filename = fields[0];
|
| - int width, height;
|
| - bool result = base::StringToInt(fields[1], &width);
|
| - LOG_ASSERT(result);
|
| - result = base::StringToInt(fields[2], &height);
|
| - LOG_ASSERT(result);
|
| - test_stream->visible_size = gfx::Size(width, height);
|
| - LOG_ASSERT(!test_stream->visible_size.IsEmpty());
|
| - int profile;
|
| - result = base::StringToInt(fields[3], &profile);
|
| - LOG_ASSERT(result);
|
| - LOG_ASSERT(profile > media::VIDEO_CODEC_PROFILE_UNKNOWN);
|
| - LOG_ASSERT(profile <= media::VIDEO_CODEC_PROFILE_MAX);
|
| - test_stream->requested_profile =
|
| - static_cast<media::VideoCodecProfile>(profile);
|
| -
|
| - if (fields.size() >= 5 && !fields[4].empty())
|
| - test_stream->out_filename = fields[4];
|
| -
|
| - if (fields.size() >= 6 && !fields[5].empty())
|
| - LOG_ASSERT(base::StringToUint(fields[5],
|
| - &test_stream->requested_bitrate));
|
| -
|
| - if (fields.size() >= 7 && !fields[6].empty())
|
| - LOG_ASSERT(base::StringToUint(fields[6],
|
| - &test_stream->requested_framerate));
|
| -
|
| - if (fields.size() >= 8 && !fields[7].empty()) {
|
| - LOG_ASSERT(base::StringToUint(fields[7],
|
| - &test_stream->requested_subsequent_bitrate));
|
| - }
|
| -
|
| - if (fields.size() >= 9 && !fields[8].empty()) {
|
| - LOG_ASSERT(base::StringToUint(fields[8],
|
| - &test_stream->requested_subsequent_framerate));
|
| - }
|
| - test_streams->push_back(test_stream);
|
| - }
|
| -}
|
| -
|
| -// Basic test environment shared across multiple test cases. We only need to
|
| -// setup it once for all test cases.
|
| -// It helps
|
| -// - maintain test stream data and other test settings.
|
| -// - clean up temporary aligned files.
|
| -// - output log to file.
|
| -class VideoEncodeAcceleratorTestEnvironment : public ::testing::Environment {
|
| - public:
|
| - VideoEncodeAcceleratorTestEnvironment(
|
| - std::unique_ptr<base::FilePath::StringType> data,
|
| - const base::FilePath& log_path,
|
| - bool run_at_fps,
|
| - bool needs_encode_latency,
|
| - bool verify_all_output)
|
| - : test_stream_data_(std::move(data)),
|
| - log_path_(log_path),
|
| - run_at_fps_(run_at_fps),
|
| - needs_encode_latency_(needs_encode_latency),
|
| - verify_all_output_(verify_all_output) {}
|
| -
|
| - virtual void SetUp() {
|
| - if (!log_path_.empty()) {
|
| - log_file_.reset(new base::File(
|
| - log_path_, base::File::FLAG_CREATE_ALWAYS | base::File::FLAG_WRITE));
|
| - LOG_ASSERT(log_file_->IsValid());
|
| - }
|
| - ParseAndReadTestStreamData(*test_stream_data_, &test_streams_);
|
| - }
|
| -
|
| - virtual void TearDown() {
|
| - for (size_t i = 0; i < test_streams_.size(); i++) {
|
| - base::DeleteFile(test_streams_[i]->aligned_in_file, false);
|
| - }
|
| - log_file_.reset();
|
| - }
|
| -
|
| - // Log one entry of machine-readable data to file and LOG(INFO).
|
| - // The log has one data entry per line in the format of "<key>: <value>".
|
| - // Note that Chrome OS video_VEAPerf autotest parses the output key and value
|
| - // pairs. Be sure to keep the autotest in sync.
|
| - void LogToFile(const std::string& key, const std::string& value) {
|
| - std::string s = base::StringPrintf("%s: %s\n", key.c_str(), value.c_str());
|
| - LOG(INFO) << s;
|
| - if (log_file_) {
|
| - log_file_->WriteAtCurrentPos(s.data(), s.length());
|
| - }
|
| - }
|
| -
|
| - // Feed the encoder with the input buffers at the requested framerate. If
|
| - // false, feed as fast as possible. This is set by the command line switch
|
| - // "--run_at_fps".
|
| - bool run_at_fps() const { return run_at_fps_; }
|
| -
|
| - // Whether to measure encode latency. This is set by the command line switch
|
| - // "--measure_latency".
|
| - bool needs_encode_latency() const { return needs_encode_latency_; }
|
| -
|
| - // Verify the encoder output of all testcases. This is set by the command line
|
| - // switch "--verify_all_output".
|
| - bool verify_all_output() const { return verify_all_output_; }
|
| -
|
| - ScopedVector<TestStream> test_streams_;
|
| -
|
| - private:
|
| - std::unique_ptr<base::FilePath::StringType> test_stream_data_;
|
| - base::FilePath log_path_;
|
| - std::unique_ptr<base::File> log_file_;
|
| - bool run_at_fps_;
|
| - bool needs_encode_latency_;
|
| - bool verify_all_output_;
|
| -};
|
| -
|
| -enum ClientState {
|
| - CS_CREATED,
|
| - CS_ENCODER_SET,
|
| - CS_INITIALIZED,
|
| - CS_ENCODING,
|
| - // Encoding has finished.
|
| - CS_FINISHED,
|
| - // Encoded frame quality has been validated.
|
| - CS_VALIDATED,
|
| - CS_ERROR,
|
| -};
|
| -
|
| -// Performs basic, codec-specific sanity checks on the stream buffers passed
|
| -// to ProcessStreamBuffer(): whether we've seen keyframes before non-keyframes,
|
| -// correct sequences of H.264 NALUs (SPS before PPS and before slices), etc.
|
| -// Calls given FrameFoundCallback when a complete frame is found while
|
| -// processing.
|
| -class StreamValidator {
|
| - public:
|
| - // To be called when a complete frame is found while processing a stream
|
| - // buffer, passing true if the frame is a keyframe. Returns false if we
|
| - // are not interested in more frames and further processing should be aborted.
|
| - typedef base::Callback<bool(bool)> FrameFoundCallback;
|
| -
|
| - virtual ~StreamValidator() {}
|
| -
|
| - // Provide a StreamValidator instance for the given |profile|.
|
| - static std::unique_ptr<StreamValidator> Create(
|
| - media::VideoCodecProfile profile,
|
| - const FrameFoundCallback& frame_cb);
|
| -
|
| - // Process and verify contents of a bitstream buffer.
|
| - virtual void ProcessStreamBuffer(const uint8_t* stream, size_t size) = 0;
|
| -
|
| - protected:
|
| - explicit StreamValidator(const FrameFoundCallback& frame_cb)
|
| - : frame_cb_(frame_cb) {}
|
| -
|
| - FrameFoundCallback frame_cb_;
|
| -};
|
| -
|
| -class H264Validator : public StreamValidator {
|
| - public:
|
| - explicit H264Validator(const FrameFoundCallback& frame_cb)
|
| - : StreamValidator(frame_cb),
|
| - seen_sps_(false),
|
| - seen_pps_(false),
|
| - seen_idr_(false) {}
|
| -
|
| - void ProcessStreamBuffer(const uint8_t* stream, size_t size) override;
|
| -
|
| - private:
|
| - // Set to true when encoder provides us with the corresponding NALU type.
|
| - bool seen_sps_;
|
| - bool seen_pps_;
|
| - bool seen_idr_;
|
| -
|
| - media::H264Parser h264_parser_;
|
| -};
|
| -
|
| -void H264Validator::ProcessStreamBuffer(const uint8_t* stream, size_t size) {
|
| - h264_parser_.SetStream(stream, size);
|
| -
|
| - while (1) {
|
| - media::H264NALU nalu;
|
| - media::H264Parser::Result result;
|
| -
|
| - result = h264_parser_.AdvanceToNextNALU(&nalu);
|
| - if (result == media::H264Parser::kEOStream)
|
| - break;
|
| -
|
| - ASSERT_EQ(media::H264Parser::kOk, result);
|
| -
|
| - bool keyframe = false;
|
| -
|
| - switch (nalu.nal_unit_type) {
|
| - case media::H264NALU::kIDRSlice:
|
| - ASSERT_TRUE(seen_sps_);
|
| - ASSERT_TRUE(seen_pps_);
|
| - seen_idr_ = true;
|
| - keyframe = true;
|
| - // fallthrough
|
| - case media::H264NALU::kNonIDRSlice: {
|
| - ASSERT_TRUE(seen_idr_);
|
| - if (!frame_cb_.Run(keyframe))
|
| - return;
|
| - break;
|
| - }
|
| -
|
| - case media::H264NALU::kSPS: {
|
| - int sps_id;
|
| - ASSERT_EQ(media::H264Parser::kOk, h264_parser_.ParseSPS(&sps_id));
|
| - seen_sps_ = true;
|
| - break;
|
| - }
|
| -
|
| - case media::H264NALU::kPPS: {
|
| - ASSERT_TRUE(seen_sps_);
|
| - int pps_id;
|
| - ASSERT_EQ(media::H264Parser::kOk, h264_parser_.ParsePPS(&pps_id));
|
| - seen_pps_ = true;
|
| - break;
|
| - }
|
| -
|
| - default:
|
| - break;
|
| - }
|
| - }
|
| -}
|
| -
|
| -class VP8Validator : public StreamValidator {
|
| - public:
|
| - explicit VP8Validator(const FrameFoundCallback& frame_cb)
|
| - : StreamValidator(frame_cb),
|
| - seen_keyframe_(false) {}
|
| -
|
| - void ProcessStreamBuffer(const uint8_t* stream, size_t size) override;
|
| -
|
| - private:
|
| - // Have we already got a keyframe in the stream?
|
| - bool seen_keyframe_;
|
| -};
|
| -
|
| -void VP8Validator::ProcessStreamBuffer(const uint8_t* stream, size_t size) {
|
| - bool keyframe = !(stream[0] & 0x01);
|
| - if (keyframe)
|
| - seen_keyframe_ = true;
|
| -
|
| - EXPECT_TRUE(seen_keyframe_);
|
| -
|
| - frame_cb_.Run(keyframe);
|
| - // TODO(posciak): We could be getting more frames in the buffer, but there is
|
| - // no simple way to detect this. We'd need to parse the frames and go through
|
| - // partition numbers/sizes. For now assume one frame per buffer.
|
| -}
|
| -
|
| -// static
|
| -std::unique_ptr<StreamValidator> StreamValidator::Create(
|
| - media::VideoCodecProfile profile,
|
| - const FrameFoundCallback& frame_cb) {
|
| - std::unique_ptr<StreamValidator> validator;
|
| -
|
| - if (IsH264(profile)) {
|
| - validator.reset(new H264Validator(frame_cb));
|
| - } else if (IsVP8(profile)) {
|
| - validator.reset(new VP8Validator(frame_cb));
|
| - } else {
|
| - LOG(FATAL) << "Unsupported profile: " << profile;
|
| - }
|
| -
|
| - return validator;
|
| -}
|
| -
|
| -class VideoFrameQualityValidator {
|
| - public:
|
| - VideoFrameQualityValidator(const media::VideoCodecProfile profile,
|
| - const base::Closure& flush_complete_cb,
|
| - const base::Closure& decode_error_cb);
|
| - void Initialize(const gfx::Size& coded_size, const gfx::Rect& visible_size);
|
| - // Save original YUV frame to compare it with the decoded frame later.
|
| - void AddOriginalFrame(scoped_refptr<media::VideoFrame> frame);
|
| - void AddDecodeBuffer(const scoped_refptr<media::DecoderBuffer>& buffer);
|
| - // Flush the decoder.
|
| - void Flush();
|
| -
|
| - private:
|
| - void InitializeCB(bool success);
|
| - void DecodeDone(media::DecodeStatus status);
|
| - void FlushDone(media::DecodeStatus status);
|
| - void VerifyOutputFrame(const scoped_refptr<media::VideoFrame>& output_frame);
|
| - void Decode();
|
| -
|
| - enum State { UNINITIALIZED, INITIALIZED, DECODING, ERROR };
|
| -
|
| - const media::VideoCodecProfile profile_;
|
| - std::unique_ptr<media::FFmpegVideoDecoder> decoder_;
|
| - media::VideoDecoder::DecodeCB decode_cb_;
|
| - // Decode callback of an EOS buffer.
|
| - media::VideoDecoder::DecodeCB eos_decode_cb_;
|
| - // Callback of Flush(). Called after all frames are decoded.
|
| - const base::Closure flush_complete_cb_;
|
| - const base::Closure decode_error_cb_;
|
| - State decoder_state_;
|
| - std::queue<scoped_refptr<media::VideoFrame>> original_frames_;
|
| - std::queue<scoped_refptr<media::DecoderBuffer>> decode_buffers_;
|
| -};
|
| -
|
| -VideoFrameQualityValidator::VideoFrameQualityValidator(
|
| - const media::VideoCodecProfile profile,
|
| - const base::Closure& flush_complete_cb,
|
| - const base::Closure& decode_error_cb)
|
| - : profile_(profile),
|
| - decoder_(new media::FFmpegVideoDecoder()),
|
| - decode_cb_(base::Bind(&VideoFrameQualityValidator::DecodeDone,
|
| - base::Unretained(this))),
|
| - eos_decode_cb_(base::Bind(&VideoFrameQualityValidator::FlushDone,
|
| - base::Unretained(this))),
|
| - flush_complete_cb_(flush_complete_cb),
|
| - decode_error_cb_(decode_error_cb),
|
| - decoder_state_(UNINITIALIZED) {
|
| - // Allow decoding of individual NALU. Entire frames are required by default.
|
| - decoder_->set_decode_nalus(true);
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::Initialize(const gfx::Size& coded_size,
|
| - const gfx::Rect& visible_size) {
|
| - media::FFmpegGlue::InitializeFFmpeg();
|
| -
|
| - gfx::Size natural_size(visible_size.size());
|
| - // The default output format of ffmpeg video decoder is YV12.
|
| - media::VideoDecoderConfig config;
|
| - if (IsVP8(profile_))
|
| - config.Initialize(media::kCodecVP8, media::VP8PROFILE_ANY, kInputFormat,
|
| - media::COLOR_SPACE_UNSPECIFIED, coded_size, visible_size,
|
| - natural_size, media::EmptyExtraData(),
|
| - media::Unencrypted());
|
| - else if (IsH264(profile_))
|
| - config.Initialize(media::kCodecH264, media::H264PROFILE_MAIN, kInputFormat,
|
| - media::COLOR_SPACE_UNSPECIFIED, coded_size, visible_size,
|
| - natural_size, media::EmptyExtraData(),
|
| - media::Unencrypted());
|
| - else
|
| - LOG_ASSERT(0) << "Invalid profile " << profile_;
|
| -
|
| - decoder_->Initialize(
|
| - config, false, nullptr,
|
| - base::Bind(&VideoFrameQualityValidator::InitializeCB,
|
| - base::Unretained(this)),
|
| - base::Bind(&VideoFrameQualityValidator::VerifyOutputFrame,
|
| - base::Unretained(this)));
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::InitializeCB(bool success) {
|
| - if (success) {
|
| - decoder_state_ = INITIALIZED;
|
| - Decode();
|
| - } else {
|
| - decoder_state_ = ERROR;
|
| - if (IsH264(profile_))
|
| - LOG(ERROR) << "Chromium does not support H264 decode. Try Chrome.";
|
| - FAIL() << "Decoder initialization error";
|
| - decode_error_cb_.Run();
|
| - }
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::AddOriginalFrame(
|
| - scoped_refptr<media::VideoFrame> frame) {
|
| - original_frames_.push(frame);
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::DecodeDone(media::DecodeStatus status) {
|
| - if (status == media::DecodeStatus::OK) {
|
| - decoder_state_ = INITIALIZED;
|
| - Decode();
|
| - } else {
|
| - decoder_state_ = ERROR;
|
| - FAIL() << "Unexpected decode status = " << status << ". Stop decoding.";
|
| - decode_error_cb_.Run();
|
| - }
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::FlushDone(media::DecodeStatus status) {
|
| - flush_complete_cb_.Run();
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::Flush() {
|
| - if (decoder_state_ != ERROR) {
|
| - decode_buffers_.push(media::DecoderBuffer::CreateEOSBuffer());
|
| - Decode();
|
| - }
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::AddDecodeBuffer(
|
| - const scoped_refptr<media::DecoderBuffer>& buffer) {
|
| - if (decoder_state_ != ERROR) {
|
| - decode_buffers_.push(buffer);
|
| - Decode();
|
| - }
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::Decode() {
|
| - if (decoder_state_ == INITIALIZED && !decode_buffers_.empty()) {
|
| - scoped_refptr<media::DecoderBuffer> next_buffer = decode_buffers_.front();
|
| - decode_buffers_.pop();
|
| - decoder_state_ = DECODING;
|
| - if (next_buffer->end_of_stream())
|
| - decoder_->Decode(next_buffer, eos_decode_cb_);
|
| - else
|
| - decoder_->Decode(next_buffer, decode_cb_);
|
| - }
|
| -}
|
| -
|
| -void VideoFrameQualityValidator::VerifyOutputFrame(
|
| - const scoped_refptr<media::VideoFrame>& output_frame) {
|
| - scoped_refptr<media::VideoFrame> original_frame = original_frames_.front();
|
| - original_frames_.pop();
|
| - gfx::Size visible_size = original_frame->visible_rect().size();
|
| -
|
| - int planes[] = {media::VideoFrame::kYPlane, media::VideoFrame::kUPlane,
|
| - media::VideoFrame::kVPlane};
|
| - double difference = 0;
|
| - for (int plane : planes) {
|
| - uint8_t* original_plane = original_frame->data(plane);
|
| - uint8_t* output_plane = output_frame->data(plane);
|
| -
|
| - size_t rows =
|
| - media::VideoFrame::Rows(plane, kInputFormat, visible_size.height());
|
| - size_t columns =
|
| - media::VideoFrame::Columns(plane, kInputFormat, visible_size.width());
|
| - size_t stride = original_frame->stride(plane);
|
| -
|
| - for (size_t i = 0; i < rows; i++)
|
| - for (size_t j = 0; j < columns; j++)
|
| - difference += std::abs(original_plane[stride * i + j] -
|
| - output_plane[stride * i + j]);
|
| - }
|
| - // Divide the difference by the size of frame.
|
| - difference /= media::VideoFrame::AllocationSize(kInputFormat, visible_size);
|
| - EXPECT_TRUE(difference <= kDecodeSimilarityThreshold)
|
| - << "differrence = " << difference << " > decode similarity threshold";
|
| -}
|
| -
|
| -class VEAClient : public VideoEncodeAccelerator::Client {
|
| - public:
|
| - VEAClient(TestStream* test_stream,
|
| - ClientStateNotification<ClientState>* note,
|
| - bool save_to_file,
|
| - unsigned int keyframe_period,
|
| - bool force_bitrate,
|
| - bool test_perf,
|
| - bool mid_stream_bitrate_switch,
|
| - bool mid_stream_framerate_switch,
|
| - bool verify_output);
|
| - ~VEAClient() override;
|
| - void CreateEncoder();
|
| - void DestroyEncoder();
|
| -
|
| - // VideoDecodeAccelerator::Client implementation.
|
| - void RequireBitstreamBuffers(unsigned int input_count,
|
| - const gfx::Size& input_coded_size,
|
| - size_t output_buffer_size) override;
|
| - void BitstreamBufferReady(int32_t bitstream_buffer_id,
|
| - size_t payload_size,
|
| - bool key_frame) override;
|
| - void NotifyError(VideoEncodeAccelerator::Error error) override;
|
| -
|
| - private:
|
| - bool has_encoder() { return encoder_.get(); }
|
| -
|
| - // Return the number of encoded frames per second.
|
| - double frames_per_second();
|
| -
|
| - std::unique_ptr<media::VideoEncodeAccelerator> CreateFakeVEA();
|
| - std::unique_ptr<media::VideoEncodeAccelerator> CreateV4L2VEA();
|
| - std::unique_ptr<media::VideoEncodeAccelerator> CreateVaapiVEA();
|
| - std::unique_ptr<media::VideoEncodeAccelerator> CreateVTVEA();
|
| -
|
| - void SetState(ClientState new_state);
|
| -
|
| - // Set current stream parameters to given |bitrate| at |framerate|.
|
| - void SetStreamParameters(unsigned int bitrate, unsigned int framerate);
|
| -
|
| - // Called when encoder is done with a VideoFrame.
|
| - void InputNoLongerNeededCallback(int32_t input_id);
|
| -
|
| - // Feed the encoder with one input frame.
|
| - void FeedEncoderWithOneInput();
|
| -
|
| - // Provide the encoder with a new output buffer.
|
| - void FeedEncoderWithOutput(base::SharedMemory* shm);
|
| -
|
| - // Called on finding a complete frame (with |keyframe| set to true for
|
| - // keyframes) in the stream, to perform codec-independent, per-frame checks
|
| - // and accounting. Returns false once we have collected all frames we needed.
|
| - bool HandleEncodedFrame(bool keyframe);
|
| -
|
| - // Verify the minimum FPS requirement.
|
| - void VerifyMinFPS();
|
| -
|
| - // Verify that stream bitrate has been close to current_requested_bitrate_,
|
| - // assuming current_framerate_ since the last time VerifyStreamProperties()
|
| - // was called. Fail the test if |force_bitrate_| is true and the bitrate
|
| - // is not within kBitrateTolerance.
|
| - void VerifyStreamProperties();
|
| -
|
| - // Log the performance data.
|
| - void LogPerf();
|
| -
|
| - // Write IVF file header to test_stream_->out_filename.
|
| - void WriteIvfFileHeader();
|
| -
|
| - // Write an IVF frame header to test_stream_->out_filename.
|
| - void WriteIvfFrameHeader(int frame_index, size_t frame_size);
|
| -
|
| - // Create and return a VideoFrame wrapping the data at |position| bytes in the
|
| - // input stream.
|
| - scoped_refptr<media::VideoFrame> CreateFrame(off_t position);
|
| -
|
| - // Prepare and return a frame wrapping the data at |position| bytes in the
|
| - // input stream, ready to be sent to encoder.
|
| - // The input frame id is returned in |input_id|.
|
| - scoped_refptr<media::VideoFrame> PrepareInputFrame(off_t position,
|
| - int32_t* input_id);
|
| -
|
| - // Update the parameters according to |mid_stream_bitrate_switch| and
|
| - // |mid_stream_framerate_switch|.
|
| - void UpdateTestStreamData(bool mid_stream_bitrate_switch,
|
| - bool mid_stream_framerate_switch);
|
| -
|
| - // Callback function of the |input_timer_|.
|
| - void OnInputTimer();
|
| -
|
| - // Called when the quality validator has decoded all the frames.
|
| - void DecodeCompleted();
|
| -
|
| - // Called when the quality validator fails to decode a frame.
|
| - void DecodeFailed();
|
| -
|
| - ClientState state_;
|
| - std::unique_ptr<VideoEncodeAccelerator> encoder_;
|
| -
|
| - TestStream* test_stream_;
|
| -
|
| - // Used to notify another thread about the state. VEAClient does not own this.
|
| - ClientStateNotification<ClientState>* note_;
|
| -
|
| - // Ids assigned to VideoFrames.
|
| - std::set<int32_t> inputs_at_client_;
|
| - int32_t next_input_id_;
|
| -
|
| - // Encode start time of all encoded frames. The position in the vector is the
|
| - // frame input id.
|
| - std::vector<base::TimeTicks> encode_start_time_;
|
| - // The encode latencies of all encoded frames. We define encode latency as the
|
| - // time delay from input of each VideoFrame (VEA::Encode()) to output of the
|
| - // corresponding BitstreamBuffer (VEA::Client::BitstreamBufferReady()).
|
| - std::vector<base::TimeDelta> encode_latencies_;
|
| -
|
| - // Ids for output BitstreamBuffers.
|
| - typedef std::map<int32_t, base::SharedMemory*> IdToSHM;
|
| - ScopedVector<base::SharedMemory> output_shms_;
|
| - IdToSHM output_buffers_at_client_;
|
| - int32_t next_output_buffer_id_;
|
| -
|
| - // Current offset into input stream.
|
| - off_t pos_in_input_stream_;
|
| - gfx::Size input_coded_size_;
|
| - // Requested by encoder.
|
| - unsigned int num_required_input_buffers_;
|
| - size_t output_buffer_size_;
|
| -
|
| - // Number of frames to encode. This may differ from the number of frames in
|
| - // stream if we need more frames for bitrate tests.
|
| - unsigned int num_frames_to_encode_;
|
| -
|
| - // Number of encoded frames we've got from the encoder thus far.
|
| - unsigned int num_encoded_frames_;
|
| -
|
| - // Frames since last bitrate verification.
|
| - unsigned int num_frames_since_last_check_;
|
| -
|
| - // True if received a keyframe while processing current bitstream buffer.
|
| - bool seen_keyframe_in_this_buffer_;
|
| -
|
| - // True if we are to save the encoded stream to a file.
|
| - bool save_to_file_;
|
| -
|
| - // Request a keyframe every keyframe_period_ frames.
|
| - const unsigned int keyframe_period_;
|
| -
|
| - // Number of keyframes requested by now.
|
| - unsigned int num_keyframes_requested_;
|
| -
|
| - // Next keyframe expected before next_keyframe_at_ + kMaxKeyframeDelay.
|
| - unsigned int next_keyframe_at_;
|
| -
|
| - // True if we are asking encoder for a particular bitrate.
|
| - bool force_bitrate_;
|
| -
|
| - // Current requested bitrate.
|
| - unsigned int current_requested_bitrate_;
|
| -
|
| - // Current expected framerate.
|
| - unsigned int current_framerate_;
|
| -
|
| - // Byte size of the encoded stream (for bitrate calculation) since last
|
| - // time we checked bitrate.
|
| - size_t encoded_stream_size_since_last_check_;
|
| -
|
| - // If true, verify performance at the end of the test.
|
| - bool test_perf_;
|
| -
|
| - // Check the output frame quality of the encoder.
|
| - bool verify_output_;
|
| -
|
| - // Used to perform codec-specific sanity checks on the stream.
|
| - std::unique_ptr<StreamValidator> stream_validator_;
|
| -
|
| - // Used to validate the encoded frame quality.
|
| - std::unique_ptr<VideoFrameQualityValidator> quality_validator_;
|
| -
|
| - // The time when the first frame is submitted for encode.
|
| - base::TimeTicks first_frame_start_time_;
|
| -
|
| - // The time when the last encoded frame is ready.
|
| - base::TimeTicks last_frame_ready_time_;
|
| -
|
| - // All methods of this class should be run on the same thread.
|
| - base::ThreadChecker thread_checker_;
|
| -
|
| - // Requested bitrate in bits per second.
|
| - unsigned int requested_bitrate_;
|
| -
|
| - // Requested initial framerate.
|
| - unsigned int requested_framerate_;
|
| -
|
| - // Bitrate to switch to in the middle of the stream.
|
| - unsigned int requested_subsequent_bitrate_;
|
| -
|
| - // Framerate to switch to in the middle of the stream.
|
| - unsigned int requested_subsequent_framerate_;
|
| -
|
| - // The timer used to feed the encoder with the input frames.
|
| - std::unique_ptr<base::RepeatingTimer> input_timer_;
|
| -};
|
| -
|
| -VEAClient::VEAClient(TestStream* test_stream,
|
| - ClientStateNotification<ClientState>* note,
|
| - bool save_to_file,
|
| - unsigned int keyframe_period,
|
| - bool force_bitrate,
|
| - bool test_perf,
|
| - bool mid_stream_bitrate_switch,
|
| - bool mid_stream_framerate_switch,
|
| - bool verify_output)
|
| - : state_(CS_CREATED),
|
| - test_stream_(test_stream),
|
| - note_(note),
|
| - next_input_id_(0),
|
| - next_output_buffer_id_(0),
|
| - pos_in_input_stream_(0),
|
| - num_required_input_buffers_(0),
|
| - output_buffer_size_(0),
|
| - num_frames_to_encode_(0),
|
| - num_encoded_frames_(0),
|
| - num_frames_since_last_check_(0),
|
| - seen_keyframe_in_this_buffer_(false),
|
| - save_to_file_(save_to_file),
|
| - keyframe_period_(keyframe_period),
|
| - num_keyframes_requested_(0),
|
| - next_keyframe_at_(0),
|
| - force_bitrate_(force_bitrate),
|
| - current_requested_bitrate_(0),
|
| - current_framerate_(0),
|
| - encoded_stream_size_since_last_check_(0),
|
| - test_perf_(test_perf),
|
| - verify_output_(verify_output),
|
| - requested_bitrate_(0),
|
| - requested_framerate_(0),
|
| - requested_subsequent_bitrate_(0),
|
| - requested_subsequent_framerate_(0) {
|
| - if (keyframe_period_)
|
| - LOG_ASSERT(kMaxKeyframeDelay < keyframe_period_);
|
| -
|
| - // Fake encoder produces an invalid stream, so skip validating it.
|
| - if (!g_fake_encoder) {
|
| - stream_validator_ = StreamValidator::Create(
|
| - test_stream_->requested_profile,
|
| - base::Bind(&VEAClient::HandleEncodedFrame, base::Unretained(this)));
|
| - CHECK(stream_validator_);
|
| - }
|
| -
|
| - if (save_to_file_) {
|
| - LOG_ASSERT(!test_stream_->out_filename.empty());
|
| - base::FilePath out_filename(test_stream_->out_filename);
|
| - // This creates or truncates out_filename.
|
| - // Without it, AppendToFile() will not work.
|
| - EXPECT_EQ(0, base::WriteFile(out_filename, NULL, 0));
|
| - }
|
| -
|
| - // Initialize the parameters of the test streams.
|
| - UpdateTestStreamData(mid_stream_bitrate_switch, mid_stream_framerate_switch);
|
| -
|
| - thread_checker_.DetachFromThread();
|
| -}
|
| -
|
| -VEAClient::~VEAClient() { LOG_ASSERT(!has_encoder()); }
|
| -
|
| -std::unique_ptr<media::VideoEncodeAccelerator> VEAClient::CreateFakeVEA() {
|
| - std::unique_ptr<media::VideoEncodeAccelerator> encoder;
|
| - if (g_fake_encoder) {
|
| - encoder.reset(new media::FakeVideoEncodeAccelerator(
|
| - scoped_refptr<base::SingleThreadTaskRunner>(
|
| - base::ThreadTaskRunnerHandle::Get())));
|
| - }
|
| - return encoder;
|
| -}
|
| -
|
| -std::unique_ptr<media::VideoEncodeAccelerator> VEAClient::CreateV4L2VEA() {
|
| - std::unique_ptr<media::VideoEncodeAccelerator> encoder;
|
| -#if defined(OS_CHROMEOS) && (defined(ARCH_CPU_ARMEL) || \
|
| - (defined(USE_OZONE) && defined(USE_V4L2_CODEC)))
|
| - scoped_refptr<V4L2Device> device = V4L2Device::Create(V4L2Device::kEncoder);
|
| - if (device)
|
| - encoder.reset(new V4L2VideoEncodeAccelerator(device));
|
| -#endif
|
| - return encoder;
|
| -}
|
| -
|
| -std::unique_ptr<media::VideoEncodeAccelerator> VEAClient::CreateVaapiVEA() {
|
| - std::unique_ptr<media::VideoEncodeAccelerator> encoder;
|
| -#if defined(OS_CHROMEOS) && defined(ARCH_CPU_X86_FAMILY)
|
| - encoder.reset(new VaapiVideoEncodeAccelerator());
|
| -#endif
|
| - return encoder;
|
| -}
|
| -
|
| -std::unique_ptr<media::VideoEncodeAccelerator> VEAClient::CreateVTVEA() {
|
| - std::unique_ptr<media::VideoEncodeAccelerator> encoder;
|
| -#if defined(OS_MACOSX)
|
| - encoder.reset(new VTVideoEncodeAccelerator());
|
| -#endif
|
| - return encoder;
|
| -}
|
| -
|
| -void VEAClient::CreateEncoder() {
|
| - DCHECK(thread_checker_.CalledOnValidThread());
|
| - LOG_ASSERT(!has_encoder());
|
| -
|
| - std::unique_ptr<media::VideoEncodeAccelerator> encoders[] = {
|
| - CreateFakeVEA(), CreateV4L2VEA(), CreateVaapiVEA(), CreateVTVEA()};
|
| -
|
| - DVLOG(1) << "Profile: " << test_stream_->requested_profile
|
| - << ", initial bitrate: " << requested_bitrate_;
|
| -
|
| - for (size_t i = 0; i < arraysize(encoders); ++i) {
|
| - if (!encoders[i])
|
| - continue;
|
| - encoder_ = std::move(encoders[i]);
|
| - SetState(CS_ENCODER_SET);
|
| - if (encoder_->Initialize(kInputFormat,
|
| - test_stream_->visible_size,
|
| - test_stream_->requested_profile,
|
| - requested_bitrate_,
|
| - this)) {
|
| - SetStreamParameters(requested_bitrate_, requested_framerate_);
|
| - SetState(CS_INITIALIZED);
|
| -
|
| - if (verify_output_ && !g_fake_encoder)
|
| - quality_validator_.reset(new VideoFrameQualityValidator(
|
| - test_stream_->requested_profile,
|
| - base::Bind(&VEAClient::DecodeCompleted, base::Unretained(this)),
|
| - base::Bind(&VEAClient::DecodeFailed, base::Unretained(this))));
|
| - return;
|
| - }
|
| - }
|
| - encoder_.reset();
|
| - LOG(ERROR) << "VideoEncodeAccelerator::Initialize() failed";
|
| - SetState(CS_ERROR);
|
| -}
|
| -
|
| -void VEAClient::DecodeCompleted() {
|
| - SetState(CS_VALIDATED);
|
| -}
|
| -
|
| -void VEAClient::DecodeFailed() {
|
| - SetState(CS_ERROR);
|
| -}
|
| -
|
| -void VEAClient::DestroyEncoder() {
|
| - DCHECK(thread_checker_.CalledOnValidThread());
|
| - if (!has_encoder())
|
| - return;
|
| - // Clear the objects that should be destroyed on the same thread as creation.
|
| - encoder_.reset();
|
| - input_timer_.reset();
|
| - quality_validator_.reset();
|
| -}
|
| -
|
| -void VEAClient::UpdateTestStreamData(bool mid_stream_bitrate_switch,
|
| - bool mid_stream_framerate_switch) {
|
| - // Use defaults for bitrate/framerate if they are not provided.
|
| - if (test_stream_->requested_bitrate == 0)
|
| - requested_bitrate_ = kDefaultBitrate;
|
| - else
|
| - requested_bitrate_ = test_stream_->requested_bitrate;
|
| -
|
| - if (test_stream_->requested_framerate == 0)
|
| - requested_framerate_ = kDefaultFramerate;
|
| - else
|
| - requested_framerate_ = test_stream_->requested_framerate;
|
| -
|
| - // If bitrate/framerate switch is requested, use the subsequent values if
|
| - // provided, or, if not, calculate them from their initial values using
|
| - // the default ratios.
|
| - // Otherwise, if a switch is not requested, keep the initial values.
|
| - if (mid_stream_bitrate_switch) {
|
| - if (test_stream_->requested_subsequent_bitrate == 0)
|
| - requested_subsequent_bitrate_ =
|
| - requested_bitrate_ * kDefaultSubsequentBitrateRatio;
|
| - else
|
| - requested_subsequent_bitrate_ =
|
| - test_stream_->requested_subsequent_bitrate;
|
| - } else {
|
| - requested_subsequent_bitrate_ = requested_bitrate_;
|
| - }
|
| - if (requested_subsequent_bitrate_ == 0)
|
| - requested_subsequent_bitrate_ = 1;
|
| -
|
| - if (mid_stream_framerate_switch) {
|
| - if (test_stream_->requested_subsequent_framerate == 0)
|
| - requested_subsequent_framerate_ =
|
| - requested_framerate_ * kDefaultSubsequentFramerateRatio;
|
| - else
|
| - requested_subsequent_framerate_ =
|
| - test_stream_->requested_subsequent_framerate;
|
| - } else {
|
| - requested_subsequent_framerate_ = requested_framerate_;
|
| - }
|
| - if (requested_subsequent_framerate_ == 0)
|
| - requested_subsequent_framerate_ = 1;
|
| -}
|
| -
|
| -double VEAClient::frames_per_second() {
|
| - LOG_ASSERT(num_encoded_frames_ != 0UL);
|
| - base::TimeDelta duration = last_frame_ready_time_ - first_frame_start_time_;
|
| - return num_encoded_frames_ / duration.InSecondsF();
|
| -}
|
| -
|
| -void VEAClient::RequireBitstreamBuffers(unsigned int input_count,
|
| - const gfx::Size& input_coded_size,
|
| - size_t output_size) {
|
| - DCHECK(thread_checker_.CalledOnValidThread());
|
| - ASSERT_EQ(state_, CS_INITIALIZED);
|
| - SetState(CS_ENCODING);
|
| -
|
| - if (quality_validator_)
|
| - quality_validator_->Initialize(input_coded_size,
|
| - gfx::Rect(test_stream_->visible_size));
|
| -
|
| - CreateAlignedInputStreamFile(input_coded_size, test_stream_);
|
| -
|
| - num_frames_to_encode_ = test_stream_->num_frames;
|
| - if (g_num_frames_to_encode > 0)
|
| - num_frames_to_encode_ = g_num_frames_to_encode;
|
| -
|
| - // We may need to loop over the stream more than once if more frames than
|
| - // provided is required for bitrate tests.
|
| - if (force_bitrate_ && num_frames_to_encode_ < kMinFramesForBitrateTests) {
|
| - DVLOG(1) << "Stream too short for bitrate test ("
|
| - << test_stream_->num_frames << " frames), will loop it to reach "
|
| - << kMinFramesForBitrateTests << " frames";
|
| - num_frames_to_encode_ = kMinFramesForBitrateTests;
|
| - }
|
| - if (save_to_file_ && IsVP8(test_stream_->requested_profile))
|
| - WriteIvfFileHeader();
|
| -
|
| - input_coded_size_ = input_coded_size;
|
| - num_required_input_buffers_ = input_count;
|
| - ASSERT_GT(num_required_input_buffers_, 0UL);
|
| -
|
| - output_buffer_size_ = output_size;
|
| - ASSERT_GT(output_buffer_size_, 0UL);
|
| -
|
| - for (unsigned int i = 0; i < kNumOutputBuffers; ++i) {
|
| - base::SharedMemory* shm = new base::SharedMemory();
|
| - LOG_ASSERT(shm->CreateAndMapAnonymous(output_buffer_size_));
|
| - output_shms_.push_back(shm);
|
| - FeedEncoderWithOutput(shm);
|
| - }
|
| -
|
| - if (g_env->run_at_fps()) {
|
| - input_timer_.reset(new base::RepeatingTimer());
|
| - input_timer_->Start(
|
| - FROM_HERE, base::TimeDelta::FromSeconds(1) / current_framerate_,
|
| - base::Bind(&VEAClient::OnInputTimer, base::Unretained(this)));
|
| - } else {
|
| - while (inputs_at_client_.size() <
|
| - num_required_input_buffers_ + kNumExtraInputFrames)
|
| - FeedEncoderWithOneInput();
|
| - }
|
| -}
|
| -
|
| -void VEAClient::BitstreamBufferReady(int32_t bitstream_buffer_id,
|
| - size_t payload_size,
|
| - bool key_frame) {
|
| - DCHECK(thread_checker_.CalledOnValidThread());
|
| - ASSERT_LE(payload_size, output_buffer_size_);
|
| -
|
| - IdToSHM::iterator it = output_buffers_at_client_.find(bitstream_buffer_id);
|
| - ASSERT_NE(it, output_buffers_at_client_.end());
|
| - base::SharedMemory* shm = it->second;
|
| - output_buffers_at_client_.erase(it);
|
| -
|
| - if (state_ == CS_FINISHED || state_ == CS_VALIDATED)
|
| - return;
|
| -
|
| - encoded_stream_size_since_last_check_ += payload_size;
|
| -
|
| - const uint8_t* stream_ptr = static_cast<const uint8_t*>(shm->memory());
|
| - if (payload_size > 0) {
|
| - if (stream_validator_) {
|
| - stream_validator_->ProcessStreamBuffer(stream_ptr, payload_size);
|
| - } else {
|
| - HandleEncodedFrame(key_frame);
|
| - }
|
| -
|
| - if (quality_validator_) {
|
| - scoped_refptr<media::DecoderBuffer> buffer(media::DecoderBuffer::CopyFrom(
|
| - reinterpret_cast<const uint8_t*>(shm->memory()),
|
| - static_cast<int>(payload_size)));
|
| - quality_validator_->AddDecodeBuffer(buffer);
|
| - // Insert EOS buffer to flush the decoder.
|
| - if (num_encoded_frames_ == num_frames_to_encode_)
|
| - quality_validator_->Flush();
|
| - }
|
| -
|
| - if (save_to_file_) {
|
| - if (IsVP8(test_stream_->requested_profile))
|
| - WriteIvfFrameHeader(num_encoded_frames_ - 1, payload_size);
|
| -
|
| - EXPECT_TRUE(base::AppendToFile(
|
| - base::FilePath::FromUTF8Unsafe(test_stream_->out_filename),
|
| - static_cast<char*>(shm->memory()),
|
| - base::checked_cast<int>(payload_size)));
|
| - }
|
| - }
|
| -
|
| - EXPECT_EQ(key_frame, seen_keyframe_in_this_buffer_);
|
| - seen_keyframe_in_this_buffer_ = false;
|
| -
|
| - FeedEncoderWithOutput(shm);
|
| -}
|
| -
|
| -void VEAClient::NotifyError(VideoEncodeAccelerator::Error error) {
|
| - DCHECK(thread_checker_.CalledOnValidThread());
|
| - SetState(CS_ERROR);
|
| -}
|
| -
|
| -void VEAClient::SetState(ClientState new_state) {
|
| - DVLOG(4) << "Changing state " << state_ << "->" << new_state;
|
| - note_->Notify(new_state);
|
| - state_ = new_state;
|
| -}
|
| -
|
| -void VEAClient::SetStreamParameters(unsigned int bitrate,
|
| - unsigned int framerate) {
|
| - current_requested_bitrate_ = bitrate;
|
| - current_framerate_ = framerate;
|
| - LOG_ASSERT(current_requested_bitrate_ > 0UL);
|
| - LOG_ASSERT(current_framerate_ > 0UL);
|
| - encoder_->RequestEncodingParametersChange(current_requested_bitrate_,
|
| - current_framerate_);
|
| - DVLOG(1) << "Switched parameters to " << current_requested_bitrate_
|
| - << " bps @ " << current_framerate_ << " FPS";
|
| -}
|
| -
|
| -void VEAClient::InputNoLongerNeededCallback(int32_t input_id) {
|
| - std::set<int32_t>::iterator it = inputs_at_client_.find(input_id);
|
| - ASSERT_NE(it, inputs_at_client_.end());
|
| - inputs_at_client_.erase(it);
|
| - if (!g_env->run_at_fps())
|
| - FeedEncoderWithOneInput();
|
| -}
|
| -
|
| -scoped_refptr<media::VideoFrame> VEAClient::CreateFrame(off_t position) {
|
| - uint8_t* frame_data_y = const_cast<uint8_t*>(
|
| - test_stream_->mapped_aligned_in_file.data() + position);
|
| - uint8_t* frame_data_u = frame_data_y + test_stream_->aligned_plane_size[0];
|
| - uint8_t* frame_data_v = frame_data_u + test_stream_->aligned_plane_size[1];
|
| - CHECK_GT(current_framerate_, 0U);
|
| -
|
| - scoped_refptr<media::VideoFrame> video_frame =
|
| - media::VideoFrame::WrapExternalYuvData(
|
| - kInputFormat, input_coded_size_,
|
| - gfx::Rect(test_stream_->visible_size), test_stream_->visible_size,
|
| - input_coded_size_.width(), input_coded_size_.width() / 2,
|
| - input_coded_size_.width() / 2, frame_data_y, frame_data_u,
|
| - frame_data_v,
|
| - base::TimeDelta().FromMilliseconds(
|
| - next_input_id_ * base::Time::kMillisecondsPerSecond /
|
| - current_framerate_));
|
| - EXPECT_NE(nullptr, video_frame.get());
|
| - return video_frame;
|
| -}
|
| -
|
| -scoped_refptr<media::VideoFrame> VEAClient::PrepareInputFrame(
|
| - off_t position,
|
| - int32_t* input_id) {
|
| - CHECK_LE(position + test_stream_->aligned_buffer_size,
|
| - test_stream_->mapped_aligned_in_file.length());
|
| -
|
| - scoped_refptr<media::VideoFrame> frame = CreateFrame(position);
|
| - EXPECT_TRUE(frame);
|
| - frame->AddDestructionObserver(
|
| - media::BindToCurrentLoop(
|
| - base::Bind(&VEAClient::InputNoLongerNeededCallback,
|
| - base::Unretained(this),
|
| - next_input_id_)));
|
| -
|
| - LOG_ASSERT(inputs_at_client_.insert(next_input_id_).second);
|
| -
|
| - *input_id = next_input_id_++;
|
| - return frame;
|
| -}
|
| -
|
| -void VEAClient::OnInputTimer() {
|
| - if (!has_encoder() || state_ != CS_ENCODING)
|
| - input_timer_.reset();
|
| - else if (inputs_at_client_.size() <
|
| - num_required_input_buffers_ + kNumExtraInputFrames)
|
| - FeedEncoderWithOneInput();
|
| - else
|
| - DVLOG(1) << "Dropping input frame";
|
| -}
|
| -
|
| -void VEAClient::FeedEncoderWithOneInput() {
|
| - if (!has_encoder() || state_ != CS_ENCODING)
|
| - return;
|
| -
|
| - size_t bytes_left =
|
| - test_stream_->mapped_aligned_in_file.length() - pos_in_input_stream_;
|
| - if (bytes_left < test_stream_->aligned_buffer_size) {
|
| - DCHECK_EQ(bytes_left, 0UL);
|
| - // Rewind if at the end of stream and we are still encoding.
|
| - // This is to flush the encoder with additional frames from the beginning
|
| - // of the stream, or if the stream is shorter that the number of frames
|
| - // we require for bitrate tests.
|
| - pos_in_input_stream_ = 0;
|
| - }
|
| -
|
| - if (quality_validator_)
|
| - quality_validator_->AddOriginalFrame(CreateFrame(pos_in_input_stream_));
|
| -
|
| - int32_t input_id;
|
| - scoped_refptr<media::VideoFrame> video_frame =
|
| - PrepareInputFrame(pos_in_input_stream_, &input_id);
|
| - pos_in_input_stream_ += test_stream_->aligned_buffer_size;
|
| -
|
| - bool force_keyframe = false;
|
| - if (keyframe_period_ && input_id % keyframe_period_ == 0) {
|
| - force_keyframe = true;
|
| - ++num_keyframes_requested_;
|
| - }
|
| -
|
| - if (input_id == 0) {
|
| - first_frame_start_time_ = base::TimeTicks::Now();
|
| - }
|
| -
|
| - if (g_env->needs_encode_latency()) {
|
| - LOG_ASSERT(input_id == static_cast<int32_t>(encode_start_time_.size()));
|
| - encode_start_time_.push_back(base::TimeTicks::Now());
|
| - }
|
| - encoder_->Encode(video_frame, force_keyframe);
|
| -}
|
| -
|
| -void VEAClient::FeedEncoderWithOutput(base::SharedMemory* shm) {
|
| - if (!has_encoder())
|
| - return;
|
| -
|
| - if (state_ != CS_ENCODING)
|
| - return;
|
| -
|
| - base::SharedMemoryHandle dup_handle;
|
| - LOG_ASSERT(shm->ShareToProcess(base::GetCurrentProcessHandle(), &dup_handle));
|
| -
|
| - media::BitstreamBuffer bitstream_buffer(
|
| - next_output_buffer_id_++, dup_handle, output_buffer_size_);
|
| - LOG_ASSERT(output_buffers_at_client_.insert(
|
| - std::make_pair(bitstream_buffer.id(), shm)).second);
|
| - encoder_->UseOutputBitstreamBuffer(bitstream_buffer);
|
| -}
|
| -
|
| -bool VEAClient::HandleEncodedFrame(bool keyframe) {
|
| - // This would be a bug in the test, which should not ignore false
|
| - // return value from this method.
|
| - LOG_ASSERT(num_encoded_frames_ <= num_frames_to_encode_);
|
| -
|
| - last_frame_ready_time_ = base::TimeTicks::Now();
|
| -
|
| - if (g_env->needs_encode_latency()) {
|
| - LOG_ASSERT(num_encoded_frames_ < encode_start_time_.size());
|
| - base::TimeTicks start_time = encode_start_time_[num_encoded_frames_];
|
| - LOG_ASSERT(!start_time.is_null());
|
| - encode_latencies_.push_back(last_frame_ready_time_ - start_time);
|
| - }
|
| -
|
| - ++num_encoded_frames_;
|
| - ++num_frames_since_last_check_;
|
| -
|
| - // Because the keyframe behavior requirements are loose, we give
|
| - // the encoder more freedom here. It could either deliver a keyframe
|
| - // immediately after we requested it, which could be for a frame number
|
| - // before the one we requested it for (if the keyframe request
|
| - // is asynchronous, i.e. not bound to any concrete frame, and because
|
| - // the pipeline can be deeper than one frame), at that frame, or after.
|
| - // So the only constraints we put here is that we get a keyframe not
|
| - // earlier than we requested one (in time), and not later than
|
| - // kMaxKeyframeDelay frames after the frame, for which we requested
|
| - // it, comes back encoded.
|
| - if (keyframe) {
|
| - if (num_keyframes_requested_ > 0) {
|
| - --num_keyframes_requested_;
|
| - next_keyframe_at_ += keyframe_period_;
|
| - }
|
| - seen_keyframe_in_this_buffer_ = true;
|
| - }
|
| -
|
| - if (num_keyframes_requested_ > 0)
|
| - EXPECT_LE(num_encoded_frames_, next_keyframe_at_ + kMaxKeyframeDelay);
|
| -
|
| - if (num_encoded_frames_ == num_frames_to_encode_ / 2) {
|
| - VerifyStreamProperties();
|
| - if (requested_subsequent_bitrate_ != current_requested_bitrate_ ||
|
| - requested_subsequent_framerate_ != current_framerate_) {
|
| - SetStreamParameters(requested_subsequent_bitrate_,
|
| - requested_subsequent_framerate_);
|
| - if (g_env->run_at_fps() && input_timer_)
|
| - input_timer_->Start(
|
| - FROM_HERE, base::TimeDelta::FromSeconds(1) / current_framerate_,
|
| - base::Bind(&VEAClient::OnInputTimer, base::Unretained(this)));
|
| - }
|
| - } else if (num_encoded_frames_ == num_frames_to_encode_) {
|
| - LogPerf();
|
| - VerifyMinFPS();
|
| - VerifyStreamProperties();
|
| - SetState(CS_FINISHED);
|
| - if (!quality_validator_)
|
| - SetState(CS_VALIDATED);
|
| - return false;
|
| - }
|
| -
|
| - return true;
|
| -}
|
| -
|
| -void VEAClient::LogPerf() {
|
| - g_env->LogToFile("Measured encoder FPS",
|
| - base::StringPrintf("%.3f", frames_per_second()));
|
| -
|
| - // Log encode latencies.
|
| - if (g_env->needs_encode_latency()) {
|
| - std::sort(encode_latencies_.begin(), encode_latencies_.end());
|
| - for (const auto& percentile : kLoggedLatencyPercentiles) {
|
| - base::TimeDelta latency = Percentile(encode_latencies_, percentile);
|
| - g_env->LogToFile(
|
| - base::StringPrintf("Encode latency for the %dth percentile",
|
| - percentile),
|
| - base::StringPrintf("%" PRId64 " us", latency.InMicroseconds()));
|
| - }
|
| - }
|
| -}
|
| -
|
| -void VEAClient::VerifyMinFPS() {
|
| - if (test_perf_)
|
| - EXPECT_GE(frames_per_second(), kMinPerfFPS);
|
| -}
|
| -
|
| -void VEAClient::VerifyStreamProperties() {
|
| - LOG_ASSERT(num_frames_since_last_check_ > 0UL);
|
| - LOG_ASSERT(encoded_stream_size_since_last_check_ > 0UL);
|
| - unsigned int bitrate = encoded_stream_size_since_last_check_ * 8 *
|
| - current_framerate_ / num_frames_since_last_check_;
|
| - DVLOG(1) << "Current chunk's bitrate: " << bitrate
|
| - << " (expected: " << current_requested_bitrate_
|
| - << " @ " << current_framerate_ << " FPS,"
|
| - << " num frames in chunk: " << num_frames_since_last_check_;
|
| -
|
| - num_frames_since_last_check_ = 0;
|
| - encoded_stream_size_since_last_check_ = 0;
|
| -
|
| - if (force_bitrate_) {
|
| - EXPECT_NEAR(bitrate,
|
| - current_requested_bitrate_,
|
| - kBitrateTolerance * current_requested_bitrate_);
|
| - }
|
| -
|
| - // All requested keyframes should've been provided. Allow the last requested
|
| - // frame to remain undelivered if we haven't reached the maximum frame number
|
| - // by which it should have arrived.
|
| - if (num_encoded_frames_ < next_keyframe_at_ + kMaxKeyframeDelay)
|
| - EXPECT_LE(num_keyframes_requested_, 1UL);
|
| - else
|
| - EXPECT_EQ(num_keyframes_requested_, 0UL);
|
| -}
|
| -
|
| -void VEAClient::WriteIvfFileHeader() {
|
| - media::IvfFileHeader header = {};
|
| -
|
| - memcpy(header.signature, media::kIvfHeaderSignature,
|
| - sizeof(header.signature));
|
| - header.version = 0;
|
| - header.header_size = sizeof(header);
|
| - header.fourcc = 0x30385056; // VP80
|
| - header.width =
|
| - base::checked_cast<uint16_t>(test_stream_->visible_size.width());
|
| - header.height =
|
| - base::checked_cast<uint16_t>(test_stream_->visible_size.height());
|
| - header.timebase_denum = requested_framerate_;
|
| - header.timebase_num = 1;
|
| - header.num_frames = num_frames_to_encode_;
|
| - header.ByteSwap();
|
| -
|
| - EXPECT_TRUE(base::AppendToFile(
|
| - base::FilePath::FromUTF8Unsafe(test_stream_->out_filename),
|
| - reinterpret_cast<char*>(&header), sizeof(header)));
|
| -}
|
| -
|
| -void VEAClient::WriteIvfFrameHeader(int frame_index, size_t frame_size) {
|
| - media::IvfFrameHeader header = {};
|
| -
|
| - header.frame_size = frame_size;
|
| - header.timestamp = frame_index;
|
| - header.ByteSwap();
|
| - EXPECT_TRUE(base::AppendToFile(
|
| - base::FilePath::FromUTF8Unsafe(test_stream_->out_filename),
|
| - reinterpret_cast<char*>(&header), sizeof(header)));
|
| -}
|
| -
|
| -// Test parameters:
|
| -// - Number of concurrent encoders. The value takes effect when there is only
|
| -// one input stream; otherwise, one encoder per input stream will be
|
| -// instantiated.
|
| -// - If true, save output to file (provided an output filename was supplied).
|
| -// - Force a keyframe every n frames.
|
| -// - Force bitrate; the actual required value is provided as a property
|
| -// of the input stream, because it depends on stream type/resolution/etc.
|
| -// - If true, measure performance.
|
| -// - If true, switch bitrate mid-stream.
|
| -// - If true, switch framerate mid-stream.
|
| -// - If true, verify the output frames of encoder.
|
| -class VideoEncodeAcceleratorTest
|
| - : public ::testing::TestWithParam<
|
| - base::Tuple<int, bool, int, bool, bool, bool, bool, bool>> {};
|
| -
|
| -TEST_P(VideoEncodeAcceleratorTest, TestSimpleEncode) {
|
| - size_t num_concurrent_encoders = base::get<0>(GetParam());
|
| - const bool save_to_file = base::get<1>(GetParam());
|
| - const unsigned int keyframe_period = base::get<2>(GetParam());
|
| - const bool force_bitrate = base::get<3>(GetParam());
|
| - const bool test_perf = base::get<4>(GetParam());
|
| - const bool mid_stream_bitrate_switch = base::get<5>(GetParam());
|
| - const bool mid_stream_framerate_switch = base::get<6>(GetParam());
|
| - const bool verify_output =
|
| - base::get<7>(GetParam()) || g_env->verify_all_output();
|
| -
|
| - ScopedVector<ClientStateNotification<ClientState> > notes;
|
| - ScopedVector<VEAClient> clients;
|
| - base::Thread encoder_thread("EncoderThread");
|
| - ASSERT_TRUE(encoder_thread.Start());
|
| -
|
| - if (g_env->test_streams_.size() > 1)
|
| - num_concurrent_encoders = g_env->test_streams_.size();
|
| -
|
| - // Create all encoders.
|
| - for (size_t i = 0; i < num_concurrent_encoders; i++) {
|
| - size_t test_stream_index = i % g_env->test_streams_.size();
|
| - // Disregard save_to_file if we didn't get an output filename.
|
| - bool encoder_save_to_file =
|
| - (save_to_file &&
|
| - !g_env->test_streams_[test_stream_index]->out_filename.empty());
|
| -
|
| - notes.push_back(new ClientStateNotification<ClientState>());
|
| - clients.push_back(new VEAClient(
|
| - g_env->test_streams_[test_stream_index], notes.back(),
|
| - encoder_save_to_file, keyframe_period, force_bitrate, test_perf,
|
| - mid_stream_bitrate_switch, mid_stream_framerate_switch, verify_output));
|
| -
|
| - encoder_thread.message_loop()->PostTask(
|
| - FROM_HERE,
|
| - base::Bind(&VEAClient::CreateEncoder,
|
| - base::Unretained(clients.back())));
|
| - }
|
| -
|
| - // All encoders must pass through states in this order.
|
| - enum ClientState state_transitions[] = {
|
| - CS_ENCODER_SET, CS_INITIALIZED, CS_ENCODING, CS_FINISHED, CS_VALIDATED};
|
| -
|
| - // Wait for all encoders to go through all states and finish.
|
| - // Do this by waiting for all encoders to advance to state n before checking
|
| - // state n+1, to verify that they are able to operate concurrently.
|
| - // It also simulates the real-world usage better, as the main thread, on which
|
| - // encoders are created/destroyed, is a single GPU Process ChildThread.
|
| - // Moreover, we can't have proper multithreading on X11, so this could cause
|
| - // hard to debug issues there, if there were multiple "ChildThreads".
|
| - for (size_t state_no = 0; state_no < arraysize(state_transitions);
|
| - ++state_no) {
|
| - for (size_t i = 0; i < num_concurrent_encoders; i++)
|
| - ASSERT_EQ(notes[i]->Wait(), state_transitions[state_no]);
|
| - }
|
| -
|
| - for (size_t i = 0; i < num_concurrent_encoders; ++i) {
|
| - encoder_thread.message_loop()->PostTask(
|
| - FROM_HERE,
|
| - base::Bind(&VEAClient::DestroyEncoder, base::Unretained(clients[i])));
|
| - }
|
| -
|
| - // This ensures all tasks have finished.
|
| - encoder_thread.Stop();
|
| -}
|
| -
|
| -#if !defined(OS_MACOSX)
|
| -INSTANTIATE_TEST_CASE_P(
|
| - SimpleEncode,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, true, 0, false, false, false, false, false),
|
| - base::MakeTuple(1, true, 0, false, false, false, false, true)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - EncoderPerf,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 0, false, true, false, false, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - ForceKeyframes,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 10, false, false, false, false, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - ForceBitrate,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 0, true, false, false, false, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - MidStreamParamSwitchBitrate,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 0, true, false, true, false, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - MidStreamParamSwitchFPS,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 0, true, false, false, true, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - MultipleEncoders,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(3, false, 0, false, false, false, false, false),
|
| - base::MakeTuple(3, false, 0, true, false, false, true, false),
|
| - base::MakeTuple(3, false, 0, true, false, true, false, false)));
|
| -#else
|
| -INSTANTIATE_TEST_CASE_P(
|
| - SimpleEncode,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, true, 0, false, false, false, false, false),
|
| - base::MakeTuple(1, true, 0, false, false, false, false, true)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - EncoderPerf,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(1, false, 0, false, true, false, false, false)));
|
| -
|
| -INSTANTIATE_TEST_CASE_P(
|
| - MultipleEncoders,
|
| - VideoEncodeAcceleratorTest,
|
| - ::testing::Values(
|
| - base::MakeTuple(3, false, 0, false, false, false, false, false)));
|
| -#endif
|
| -
|
| -// TODO(posciak): more tests:
|
| -// - async FeedEncoderWithOutput
|
| -// - out-of-order return of outputs to encoder
|
| -// - multiple encoders + decoders
|
| -// - mid-stream encoder_->Destroy()
|
| -
|
| -} // namespace
|
| -} // namespace content
|
| -
|
| -int main(int argc, char** argv) {
|
| - testing::InitGoogleTest(&argc, argv); // Removes gtest-specific args.
|
| - base::CommandLine::Init(argc, argv);
|
| -
|
| - base::ShadowingAtExitManager at_exit_manager;
|
| - base::MessageLoop main_loop;
|
| -
|
| - std::unique_ptr<base::FilePath::StringType> test_stream_data(
|
| - new base::FilePath::StringType(
|
| - media::GetTestDataFilePath(content::g_default_in_filename).value() +
|
| - content::g_default_in_parameters));
|
| -
|
| - // Needed to enable DVLOG through --vmodule.
|
| - logging::LoggingSettings settings;
|
| - settings.logging_dest = logging::LOG_TO_SYSTEM_DEBUG_LOG;
|
| - LOG_ASSERT(logging::InitLogging(settings));
|
| -
|
| - const base::CommandLine* cmd_line = base::CommandLine::ForCurrentProcess();
|
| - DCHECK(cmd_line);
|
| -
|
| - bool run_at_fps = false;
|
| - bool needs_encode_latency = false;
|
| - bool verify_all_output = false;
|
| - base::FilePath log_path;
|
| -
|
| - base::CommandLine::SwitchMap switches = cmd_line->GetSwitches();
|
| - for (base::CommandLine::SwitchMap::const_iterator it = switches.begin();
|
| - it != switches.end();
|
| - ++it) {
|
| - if (it->first == "test_stream_data") {
|
| - test_stream_data->assign(it->second.c_str());
|
| - continue;
|
| - }
|
| - // Output machine-readable logs with fixed formats to a file.
|
| - if (it->first == "output_log") {
|
| - log_path = base::FilePath(
|
| - base::FilePath::StringType(it->second.begin(), it->second.end()));
|
| - continue;
|
| - }
|
| - if (it->first == "num_frames_to_encode") {
|
| - std::string input(it->second.begin(), it->second.end());
|
| - LOG_ASSERT(base::StringToInt(input, &content::g_num_frames_to_encode));
|
| - continue;
|
| - }
|
| - if (it->first == "measure_latency") {
|
| - needs_encode_latency = true;
|
| - continue;
|
| - }
|
| - if (it->first == "fake_encoder") {
|
| - content::g_fake_encoder = true;
|
| - continue;
|
| - }
|
| - if (it->first == "run_at_fps") {
|
| - run_at_fps = true;
|
| - continue;
|
| - }
|
| - if (it->first == "verify_all_output") {
|
| - verify_all_output = true;
|
| - continue;
|
| - }
|
| - if (it->first == "v" || it->first == "vmodule")
|
| - continue;
|
| - if (it->first == "ozone-platform" || it->first == "ozone-use-surfaceless")
|
| - continue;
|
| - LOG(FATAL) << "Unexpected switch: " << it->first << ":" << it->second;
|
| - }
|
| -
|
| - if (needs_encode_latency && !run_at_fps) {
|
| - // Encode latency can only be measured with --run_at_fps. Otherwise, we get
|
| - // skewed results since it may queue too many frames at once with the same
|
| - // encode start time.
|
| - LOG(FATAL) << "--measure_latency requires --run_at_fps enabled to work.";
|
| - }
|
| -
|
| -#if defined(OS_CHROMEOS) && defined(ARCH_CPU_X86_FAMILY)
|
| - content::VaapiWrapper::PreSandboxInitialization();
|
| -#endif
|
| -
|
| - content::g_env =
|
| - reinterpret_cast<content::VideoEncodeAcceleratorTestEnvironment*>(
|
| - testing::AddGlobalTestEnvironment(
|
| - new content::VideoEncodeAcceleratorTestEnvironment(
|
| - std::move(test_stream_data), log_path, run_at_fps,
|
| - needs_encode_latency, verify_all_output)));
|
| -
|
| - return RUN_ALL_TESTS();
|
| -}
|
|
|