blob: de8119aff6809679c651244b6f6a8dec678c9ee9 [file] [log] [blame]
// Copyright 2019 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.
// This has to be included first.
// See http://code.google.com/p/googletest/issues/detail?id=371
#include "testing/gtest/include/gtest/gtest.h"
#include <unistd.h>
#include <map>
#include <vector>
#include <va/va.h>
#include <va/va_str.h>
#include "base/callback_helpers.h"
#include "base/containers/contains.h"
#include "base/cpu.h"
#include "base/files/file.h"
#include "base/files/scoped_file.h"
#include "base/logging.h"
#include "base/process/launch.h"
#include "base/strings/pattern.h"
#include "base/strings/string_split.h"
#include "base/strings/string_util.h"
#include "base/test/launcher/unit_test_launcher.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/test_suite.h"
#include "build/chromeos_buildflags.h"
#include "media/base/media_switches.h"
#include "media/gpu/vaapi/vaapi_wrapper.h"
#include "media/media_buildflags.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
namespace media {
namespace {
absl::optional<VAProfile> ConvertToVAProfile(VideoCodecProfile profile) {
// A map between VideoCodecProfile and VAProfile.
const std::map<VideoCodecProfile, VAProfile> kProfileMap = {
// VAProfileH264Baseline is deprecated in <va/va.h> from libva 2.0.0.
{H264PROFILE_BASELINE, VAProfileH264ConstrainedBaseline},
{H264PROFILE_MAIN, VAProfileH264Main},
{H264PROFILE_HIGH, VAProfileH264High},
{VP8PROFILE_ANY, VAProfileVP8Version0_3},
{VP9PROFILE_PROFILE0, VAProfileVP9Profile0},
{VP9PROFILE_PROFILE2, VAProfileVP9Profile2},
{AV1PROFILE_PROFILE_MAIN, VAProfileAV1Profile0},
#if BUILDFLAG(ENABLE_PLATFORM_HEVC_DECODING)
{HEVCPROFILE_MAIN, VAProfileHEVCMain},
{HEVCPROFILE_MAIN10, VAProfileHEVCMain10},
#endif
};
auto it = kProfileMap.find(profile);
return it != kProfileMap.end() ? absl::make_optional<VAProfile>(it->second)
: absl::nullopt;
}
// Converts the given string to VAProfile
absl::optional<VAProfile> StringToVAProfile(const std::string& va_profile) {
const std::map<std::string, VAProfile> kStringToVAProfile = {
{"VAProfileNone", VAProfileNone},
{"VAProfileH264ConstrainedBaseline", VAProfileH264ConstrainedBaseline},
// Even though it's deprecated, we leave VAProfileH264Baseline's
// translation here to assert we never encounter it.
{"VAProfileH264Baseline", VAProfileH264Baseline},
{"VAProfileH264Main", VAProfileH264Main},
{"VAProfileH264High", VAProfileH264High},
{"VAProfileJPEGBaseline", VAProfileJPEGBaseline},
{"VAProfileVP8Version0_3", VAProfileVP8Version0_3},
{"VAProfileVP9Profile0", VAProfileVP9Profile0},
{"VAProfileVP9Profile2", VAProfileVP9Profile2},
{"VAProfileAV1Profile0", VAProfileAV1Profile0},
#if BUILDFLAG(ENABLE_PLATFORM_HEVC_DECODING)
{"VAProfileHEVCMain", VAProfileHEVCMain},
{"VAProfileHEVCMain10", VAProfileHEVCMain10},
#endif
#if BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
{"VAProfileProtected", VAProfileProtected},
#endif // BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
};
auto it = kStringToVAProfile.find(va_profile);
return it != kStringToVAProfile.end()
? absl::make_optional<VAProfile>(it->second)
: absl::nullopt;
}
// Converts the given string to VAEntrypoint
absl::optional<VAEntrypoint> StringToVAEntrypoint(
const std::string& va_entrypoint) {
const std::map<std::string, VAEntrypoint> kStringToVAEntrypoint = {
{"VAEntrypointVLD", VAEntrypointVLD},
{"VAEntrypointEncSlice", VAEntrypointEncSlice},
{"VAEntrypointEncPicture", VAEntrypointEncPicture},
{"VAEntrypointEncSliceLP", VAEntrypointEncSliceLP},
{"VAEntrypointVideoProc", VAEntrypointVideoProc},
#if BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
{"VAEntrypointProtectedContent", VAEntrypointProtectedContent},
#endif // BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
};
auto it = kStringToVAEntrypoint.find(va_entrypoint);
return it != kStringToVAEntrypoint.end()
? absl::make_optional<VAEntrypoint>(it->second)
: absl::nullopt;
}
std::unique_ptr<base::test::ScopedFeatureList> CreateScopedFeatureList() {
auto scoped_feature_list = std::make_unique<base::test::ScopedFeatureList>();
scoped_feature_list->InitWithFeatures(
/*enabled_features=*/{media::kVaapiAV1Decoder},
/*disabled_features=*/{});
return scoped_feature_list;
}
unsigned int ToVaRTFormat(uint32_t va_fourcc) {
switch (va_fourcc) {
case VA_FOURCC_I420:
return VA_RT_FORMAT_YUV420;
case VA_FOURCC_YUY2:
return VA_RT_FORMAT_YUV422;
case VA_FOURCC_RGBA:
return VA_RT_FORMAT_RGB32;
case VA_FOURCC_P010:
return VA_RT_FORMAT_YUV420_10;
}
return kInvalidVaRtFormat;
}
} // namespace
class VaapiTest : public testing::Test {
public:
VaapiTest() : scoped_feature_list_(CreateScopedFeatureList()) {}
~VaapiTest() override = default;
private:
std::unique_ptr<base::test::ScopedFeatureList> scoped_feature_list_;
};
std::map<VAProfile, std::vector<VAEntrypoint>> ParseVainfo(
const std::string& output) {
const std::vector<std::string> lines =
base::SplitString(output, "\n", base::WhitespaceHandling::TRIM_WHITESPACE,
base::SplitResult::SPLIT_WANT_ALL);
std::map<VAProfile, std::vector<VAEntrypoint>> info;
for (const std::string& line : lines) {
if (!base::StartsWith(line, "VAProfile",
base::CompareCase::INSENSITIVE_ASCII)) {
continue;
}
std::vector<std::string> res =
base::SplitString(line, ":", base::WhitespaceHandling::TRIM_WHITESPACE,
base::SplitResult::SPLIT_WANT_ALL);
if (res.size() != 2) {
LOG(ERROR) << "Unexpected line: " << line;
continue;
}
auto va_profile = StringToVAProfile(res[0]);
if (!va_profile)
continue;
auto va_entrypoint = StringToVAEntrypoint(res[1]);
if (!va_entrypoint)
continue;
info[*va_profile].push_back(*va_entrypoint);
DVLOG(3) << line;
}
return info;
}
std::map<VAProfile, std::vector<VAEntrypoint>> RetrieveVAInfoOutput() {
int fds[2];
PCHECK(pipe(fds) == 0);
base::File read_pipe(fds[0]);
base::ScopedFD write_pipe_fd(fds[1]);
base::LaunchOptions options;
options.fds_to_remap.emplace_back(write_pipe_fd.get(), STDOUT_FILENO);
std::vector<std::string> argv = {"vainfo"};
EXPECT_TRUE(LaunchProcess(argv, options).IsValid());
write_pipe_fd.reset();
char buf[4096] = {};
int n = read_pipe.ReadAtCurrentPos(buf, sizeof(buf));
PCHECK(n >= 0);
EXPECT_LT(n, 4096);
std::string output(buf, n);
DVLOG(4) << output;
return ParseVainfo(output);
}
TEST_F(VaapiTest, VaapiSandboxInitialization) {
// Here we just test that the PreSandboxInitialization() in SetUp() worked
// fine. Said initialization is buried in internal singletons, but we can
// verify that at least the implementation type has been filled in.
EXPECT_NE(VaapiWrapper::GetImplementationType(), VAImplementation::kInvalid);
}
// Commit [1] deprecated VAProfileH264Baseline from libva in 2017 (release
// 2.0.0). This test verifies that such profile is never seen in the lab.
// [1] https://github.com/intel/libva/commit/6f69256f8ccc9a73c0b196ab77ac69ab1f4f33c2
TEST_F(VaapiTest, VerifyNoVAProfileH264Baseline) {
const auto va_info = RetrieveVAInfoOutput();
EXPECT_FALSE(base::Contains(va_info, VAProfileH264Baseline));
}
// Verifies that every VAProfile from VaapiWrapper::GetSupportedDecodeProfiles()
// is indeed supported by the command line vainfo utility and by
// VaapiWrapper::IsDecodeSupported().
TEST_F(VaapiTest, GetSupportedDecodeProfiles) {
const auto va_info = RetrieveVAInfoOutput();
for (const auto& profile : VaapiWrapper::GetSupportedDecodeProfiles()) {
const auto va_profile = ConvertToVAProfile(profile.profile);
ASSERT_TRUE(va_profile.has_value());
EXPECT_TRUE(base::Contains(va_info.at(*va_profile), VAEntrypointVLD))
<< " profile: " << GetProfileName(profile.profile)
<< ", va profile: " << vaProfileStr(*va_profile);
EXPECT_TRUE(VaapiWrapper::IsDecodeSupported(*va_profile))
<< " profile: " << GetProfileName(profile.profile)
<< ", va profile: " << vaProfileStr(*va_profile);
}
}
// Verifies that every VAProfile from VaapiWrapper::GetSupportedEncodeProfiles()
// is indeed supported by the command line vainfo utility.
TEST_F(VaapiTest, GetSupportedEncodeProfiles) {
const auto va_info = RetrieveVAInfoOutput();
for (const auto& profile : VaapiWrapper::GetSupportedEncodeProfiles()) {
const auto va_profile = ConvertToVAProfile(profile.profile);
ASSERT_TRUE(va_profile.has_value());
EXPECT_TRUE(base::Contains(va_info.at(*va_profile), VAEntrypointEncSlice) ||
base::Contains(va_info.at(*va_profile), VAEntrypointEncSliceLP))
<< " profile: " << GetProfileName(profile.profile)
<< ", va profile: " << vaProfileStr(*va_profile);
}
}
#if BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
// Verifies that VAProfileProtected is indeed supported by the command line
// vainfo utility.
TEST_F(VaapiTest, VaapiProfileProtected) {
const auto va_info = RetrieveVAInfoOutput();
EXPECT_TRUE(base::Contains(va_info.at(VAProfileProtected),
VAEntrypointProtectedContent))
<< ", va profile: " << vaProfileStr(VAProfileProtected);
}
#endif // BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
// Verifies that if JPEG decoding and encoding are supported by VaapiWrapper,
// they are also supported by by the command line vainfo utility.
TEST_F(VaapiTest, VaapiProfilesJPEG) {
const auto va_info = RetrieveVAInfoOutput();
EXPECT_EQ(VaapiWrapper::IsDecodeSupported(VAProfileJPEGBaseline),
base::Contains(va_info.at(VAProfileJPEGBaseline), VAEntrypointVLD));
EXPECT_EQ(VaapiWrapper::IsJpegEncodeSupported(),
base::Contains(va_info.at(VAProfileJPEGBaseline),
VAEntrypointEncPicture));
}
// Verifies that the default VAEntrypoint as per VaapiWrapper is indeed among
// the supported ones.
TEST_F(VaapiTest, DefaultEntrypointIsSupported) {
for (size_t i = 0; i < VaapiWrapper::kCodecModeMax; ++i) {
const auto wrapper_mode = static_cast<VaapiWrapper::CodecMode>(i);
std::map<VAProfile, std::vector<VAEntrypoint>> configurations =
VaapiWrapper::GetSupportedConfigurationsForCodecModeForTesting(
wrapper_mode);
for (const auto& profile_and_entrypoints : configurations) {
const VAEntrypoint default_entrypoint =
VaapiWrapper::GetDefaultVaEntryPoint(wrapper_mode,
profile_and_entrypoints.first);
const auto& supported_entrypoints = profile_and_entrypoints.second;
EXPECT_TRUE(base::Contains(supported_entrypoints, default_entrypoint))
<< "Default VAEntrypoint " << vaEntrypointStr(default_entrypoint)
<< " (VaapiWrapper mode = " << wrapper_mode
<< ") is not supported for "
<< vaProfileStr(profile_and_entrypoints.first);
}
}
}
// Verifies that VaapiWrapper::CreateContext() will queue up a buffer to set the
// encoder to its lowest quality setting if a given VAProfile and VAEntrypoint
// claims to support configuring it.
TEST_F(VaapiTest, LowQualityEncodingSetting) {
// This test only applies to low powered Intel processors.
constexpr int kPentiumAndLaterFamily = 0x06;
const base::CPU cpuid;
const bool is_core_y_processor =
base::MatchPattern(cpuid.cpu_brand(), "Intel(R) Core(TM) *Y CPU*");
const bool is_low_power_intel =
cpuid.family() == kPentiumAndLaterFamily &&
(base::Contains(cpuid.cpu_brand(), "Pentium") ||
base::Contains(cpuid.cpu_brand(), "Celeron") || is_core_y_processor);
if (!is_low_power_intel)
GTEST_SKIP() << "Not an Intel low power processor";
std::map<VAProfile, std::vector<VAEntrypoint>> configurations =
VaapiWrapper::GetSupportedConfigurationsForCodecModeForTesting(
VaapiWrapper::kEncodeConstantBitrate);
for (const auto& codec_mode :
{VaapiWrapper::kEncodeConstantBitrate,
VaapiWrapper::kEncodeConstantQuantizationParameter}) {
std::map<VAProfile, std::vector<VAEntrypoint>> configurations =
VaapiWrapper::GetSupportedConfigurationsForCodecModeForTesting(
codec_mode);
for (const auto& profile_and_entrypoints : configurations) {
const VAProfile va_profile = profile_and_entrypoints.first;
scoped_refptr<VaapiWrapper> wrapper = VaapiWrapper::Create(
VaapiWrapper::kEncodeConstantBitrate, va_profile,
EncryptionScheme::kUnencrypted, base::DoNothing());
// Depending on the GPU Gen, flags and policies, we may or may not utilize
// all entrypoints (e.g. we might always want VAEntrypointEncSliceLP if
// supported and enabled). Query VaapiWrapper's mandated entry point.
const VAEntrypoint entrypoint =
VaapiWrapper::GetDefaultVaEntryPoint(codec_mode, va_profile);
ASSERT_TRUE(base::Contains(profile_and_entrypoints.second, entrypoint));
VAConfigAttrib attrib{};
attrib.type = VAConfigAttribEncQualityRange;
{
base::AutoLock auto_lock(*wrapper->va_lock_);
VAStatus va_res = vaGetConfigAttributes(
wrapper->va_display_, va_profile, entrypoint, &attrib, 1);
ASSERT_EQ(va_res, VA_STATUS_SUCCESS);
}
const auto quality_level = attrib.value;
if (quality_level == VA_ATTRIB_NOT_SUPPORTED || quality_level <= 1u)
continue;
DLOG(INFO) << vaProfileStr(va_profile)
<< " supports encoding quality setting, with max value "
<< quality_level;
// If we get here it means the |va_profile| and |entrypoint| support
// the quality setting. We cannot inspect what the driver does with this
// number (it could ignore it), so instead just make sure there's a
// |pending_va_buffers_| that, when mapped, looks correct. That buffer
// should be created by CreateContext().
ASSERT_TRUE(wrapper->CreateContext(gfx::Size(640, 368)));
ASSERT_EQ(wrapper->pending_va_buffers_.size(), 1u);
{
base::AutoLock auto_lock(*wrapper->va_lock_);
ScopedVABufferMapping mapping(wrapper->va_lock_, wrapper->va_display_,
wrapper->pending_va_buffers_.front());
ASSERT_TRUE(mapping.IsValid());
auto* const va_buffer =
reinterpret_cast<VAEncMiscParameterBuffer*>(mapping.data());
EXPECT_EQ(va_buffer->type, VAEncMiscParameterTypeQualityLevel);
auto* const enc_quality =
reinterpret_cast<VAEncMiscParameterBufferQualityLevel*>(
va_buffer->data);
EXPECT_EQ(enc_quality->quality_level, quality_level)
<< vaProfileStr(va_profile) << " " << vaEntrypointStr(entrypoint);
}
}
}
}
class VaapiVppTest
: public VaapiTest,
public testing::WithParamInterface<std::tuple<uint32_t, uint32_t>> {
public:
VaapiVppTest() = default;
~VaapiVppTest() override = default;
// Populate meaningful test suffixes instead of /0, /1, etc.
struct PrintToStringParamName {
template <class ParamType>
std::string operator()(
const testing::TestParamInfo<ParamType>& info) const {
std::stringstream ss;
ss << FourccToString(std::get<0>(info.param)) << "_to_"
<< FourccToString(std::get<1>(info.param));
return ss.str();
}
};
};
TEST_P(VaapiVppTest, BlitWithVAAllocatedSurfaces) {
const uint32_t va_fourcc_in = std::get<0>(GetParam());
const uint32_t va_fourcc_out = std::get<1>(GetParam());
// TODO(b/187852384): enable the other two backends.
if (VaapiWrapper::GetImplementationType() != VAImplementation::kIntelIHD)
GTEST_SKIP() << "backend not supported";
if (!VaapiWrapper::IsVppFormatSupported(va_fourcc_in) ||
!VaapiWrapper::IsVppFormatSupported(va_fourcc_out)) {
GTEST_SKIP() << FourccToString(va_fourcc_in) << " -> "
<< FourccToString(va_fourcc_out) << " not supported";
}
constexpr gfx::Size kInputSize(640, 320);
constexpr gfx::Size kOutputSize(320, 180);
ASSERT_TRUE(VaapiWrapper::IsVppResolutionAllowed(kInputSize));
ASSERT_TRUE(VaapiWrapper::IsVppResolutionAllowed(kOutputSize));
auto wrapper =
VaapiWrapper::Create(VaapiWrapper::kVideoProcess, VAProfileNone,
EncryptionScheme::kUnencrypted, base::DoNothing());
ASSERT_TRUE(!!wrapper);
// Size is unnecessary for a VPP context.
ASSERT_TRUE(wrapper->CreateContext(gfx::Size()));
const unsigned int va_rt_format_in = ToVaRTFormat(va_fourcc_in);
ASSERT_NE(va_rt_format_in, kInvalidVaRtFormat);
const unsigned int va_rt_format_out = ToVaRTFormat(va_fourcc_out);
ASSERT_NE(va_rt_format_out, kInvalidVaRtFormat);
auto scoped_surfaces = wrapper->CreateScopedVASurfaces(
va_rt_format_in, kInputSize, {VaapiWrapper::SurfaceUsageHint::kGeneric},
1u, /*visible_size=*/absl::nullopt, /*va_fourcc=*/absl::nullopt);
ASSERT_FALSE(scoped_surfaces.empty());
std::unique_ptr<ScopedVASurface> scoped_surface_in =
std::move(scoped_surfaces[0]);
scoped_surfaces = wrapper->CreateScopedVASurfaces(
va_rt_format_out, kOutputSize, {VaapiWrapper::SurfaceUsageHint::kGeneric},
1u, /*visible_size=*/absl::nullopt, /*va_fourcc=*/absl::nullopt);
ASSERT_FALSE(scoped_surfaces.empty());
std::unique_ptr<ScopedVASurface> scoped_surface_out =
std::move(scoped_surfaces[0]);
scoped_refptr<VASurface> surface_in = base::MakeRefCounted<VASurface>(
scoped_surface_in->id(), kInputSize, va_rt_format_in, base::DoNothing());
scoped_refptr<VASurface> surface_out =
base::MakeRefCounted<VASurface>(scoped_surface_out->id(), kOutputSize,
va_rt_format_out, base::DoNothing());
ASSERT_TRUE(wrapper->BlitSurface(*surface_in, *surface_out,
gfx::Rect(kInputSize),
gfx::Rect(kOutputSize), VIDEO_ROTATION_0));
ASSERT_TRUE(wrapper->SyncSurface(scoped_surface_out->id()));
wrapper->DestroyContext();
}
// TODO(b/187852384): Consider adding more VaapiVppTest cases, e.g. crops.
// Note: vaCreateSurfaces() uses the RT version of the Four CC, so we don't need
// to consider swizzlings, since they'll end up mapped to the same RT format.
constexpr uint32_t kVAFourCCs[] = {VA_FOURCC_I420, VA_FOURCC_YUY2,
VA_FOURCC_RGBA, VA_FOURCC_P010};
INSTANTIATE_TEST_SUITE_P(,
VaapiVppTest,
::testing::Combine(::testing::ValuesIn(kVAFourCCs),
::testing::ValuesIn(kVAFourCCs)),
VaapiVppTest::PrintToStringParamName());
} // namespace media
int main(int argc, char** argv) {
base::TestSuite test_suite(argc, argv);
{
// Enables/Disables features during PreSandboxInitialization(). We have to
// destruct ScopedFeatureList after it because base::TestSuite::Run()
// creates a ScopedFeatureList and multiple concurrent ScopedFeatureLists
// are not allowed.
auto scoped_feature_list = media::CreateScopedFeatureList();
// PreSandboxInitialization() loads and opens the driver, queries its
// capabilities and fills in the VASupportedProfiles.
media::VaapiWrapper::PreSandboxInitialization();
}
return base::LaunchUnitTests(
argc, argv,
base::BindOnce(&base::TestSuite::Run, base::Unretained(&test_suite)));
}