blob: bcaa6a20e82d989bfa3dded4214ab404be67678c [file] [log] [blame]
// Copyright 2023 Google LLC
// SPDX-License-Identifier: BSD-2-Clause
#include "avif/avif.h"
#include "aviftest_helpers.h"
#include "gtest/gtest.h"
using ::testing::Combine;
using ::testing::Values;
namespace libavif {
namespace {
//------------------------------------------------------------------------------
class AvifCondensedImageBoxTest
: public testing::TestWithParam<std::tuple<
/*width=*/int, /*height=*/int, /*depth=*/int, avifPixelFormat,
avifPlanesFlags, avifRange, /*create_icc=*/bool, /*create_exif=*/bool,
/*create_xmp=*/bool, /*create_clli=*/bool, avifTransformFlags>> {};
TEST_P(AvifCondensedImageBoxTest, SimpleOpaque) {
const int width = std::get<0>(GetParam());
const int height = std::get<1>(GetParam());
const int depth = std::get<2>(GetParam());
const avifPixelFormat format = std::get<3>(GetParam());
const avifPlanesFlags planes = std::get<4>(GetParam());
const avifRange range = std::get<5>(GetParam());
const bool create_icc = std::get<6>(GetParam());
const bool create_exif = std::get<7>(GetParam());
const bool create_xmp = std::get<8>(GetParam());
const bool create_clli = std::get<9>(GetParam());
const avifTransformFlags create_transform_flags = std::get<10>(GetParam());
testutil::AvifImagePtr image =
testutil::CreateImage(width, height, depth, format, planes, range);
ASSERT_NE(image, nullptr);
testutil::FillImageGradient(image.get()); // The pixels do not matter.
if (create_icc) {
avifImageSetProfileICC(image.get(), testutil::kSampleIcc.data(),
testutil::kSampleIcc.size());
}
if (create_exif) {
size_t exif_tiff_header_offset; // Must be 0 for 'avir' brand.
ASSERT_EQ(avifGetExifTiffHeaderOffset(testutil::kSampleExif.data(),
testutil::kSampleExif.size(),
&exif_tiff_header_offset),
AVIF_RESULT_OK);
avifImageSetMetadataExif(
image.get(), testutil::kSampleExif.data() + exif_tiff_header_offset,
testutil::kSampleExif.size() - exif_tiff_header_offset);
}
if (create_xmp) {
avifImageSetMetadataXMP(image.get(), testutil::kSampleXmp.data(),
testutil::kSampleXmp.size());
}
if (create_clli) {
image->clli.maxCLL = 1;
image->clli.maxPALL = 2;
}
image->transformFlags = create_transform_flags;
if (create_transform_flags & AVIF_TRANSFORM_PASP) {
image->pasp.hSpacing = 1;
image->pasp.vSpacing = 2;
}
if (create_transform_flags & AVIF_TRANSFORM_CLAP) {
const avifCropRect rect{image->width / 4, image->height / 4,
std::min(1u, image->width / 2),
std::min(1u, image->height / 2)};
avifDiagnostics diag;
ASSERT_TRUE(avifCleanApertureBoxConvertCropRect(&image->clap, &rect,
image->width, image->height,
image->yuvFormat, &diag));
}
if (create_transform_flags & AVIF_TRANSFORM_IROT) {
image->irot.angle = 2;
}
if (create_transform_flags & AVIF_TRANSFORM_IMIR) {
image->imir.axis = 1;
}
// Encode.
testutil::AvifRwData encoded_coni;
testutil::AvifEncoderPtr encoder(avifEncoderCreate(), avifEncoderDestroy);
ASSERT_NE(encoder, nullptr);
encoder->speed = AVIF_SPEED_FASTEST;
encoder->headerFormat = AVIF_HEADER_REDUCED;
ASSERT_EQ(avifEncoderWrite(encoder.get(), image.get(), &encoded_coni),
AVIF_RESULT_OK);
// Decode.
const testutil::AvifImagePtr decoded_coni =
testutil::Decode(encoded_coni.data, encoded_coni.size);
ASSERT_NE(decoded_coni, nullptr);
// Compare.
testutil::AvifRwData encoded_meta =
testutil::Encode(image.get(), encoder->speed);
ASSERT_NE(encoded_meta.data, nullptr);
// At least 200 bytes should be saved.
EXPECT_LT(encoded_coni.size, encoded_meta.size - 200);
const testutil::AvifImagePtr decoded_meta =
testutil::Decode(encoded_meta.data, encoded_meta.size);
ASSERT_NE(decoded_meta, nullptr);
// Only the container changed. The pixels, features and metadata should be
// identical.
EXPECT_TRUE(
testutil::AreImagesEqual(*decoded_meta.get(), *decoded_coni.get()));
}
INSTANTIATE_TEST_SUITE_P(OnePixel, AvifCondensedImageBoxTest,
Combine(/*width=*/Values(1), /*height=*/Values(1),
/*depth=*/Values(8),
Values(AVIF_PIXEL_FORMAT_YUV444),
Values(AVIF_PLANES_YUV, AVIF_PLANES_ALL),
Values(AVIF_RANGE_LIMITED, AVIF_RANGE_FULL),
/*create_icc=*/Values(false, true),
/*create_exif=*/Values(false, true),
/*create_xmp=*/Values(false, true),
/*create_clli=*/Values(false),
Values(AVIF_TRANSFORM_NONE)));
INSTANTIATE_TEST_SUITE_P(
DepthsSubsamplings, AvifCondensedImageBoxTest,
Combine(/*width=*/Values(12), /*height=*/Values(34),
/*depth=*/Values(8, 10, 12),
Values(AVIF_PIXEL_FORMAT_YUV444, AVIF_PIXEL_FORMAT_YUV420,
AVIF_PIXEL_FORMAT_YUV400),
Values(AVIF_PLANES_ALL), Values(AVIF_RANGE_FULL),
/*create_icc=*/Values(false), /*create_exif=*/Values(false),
/*create_xmp=*/Values(false), /*create_clli=*/Values(false),
Values(AVIF_TRANSFORM_NONE)));
INSTANTIATE_TEST_SUITE_P(
Dimensions, AvifCondensedImageBoxTest,
Combine(/*width=*/Values(127), /*height=*/Values(200), /*depth=*/Values(8),
Values(AVIF_PIXEL_FORMAT_YUV444), Values(AVIF_PLANES_ALL),
Values(AVIF_RANGE_FULL), /*create_icc=*/Values(true),
/*create_exif=*/Values(true), /*create_xmp=*/Values(true),
/*create_clli=*/Values(false), Values(AVIF_TRANSFORM_NONE)));
// Use CLLI and transform properties that are not supported by a plain
// CondensedImageBox to force the use of an ExtendedMetaBox.
INSTANTIATE_TEST_SUITE_P(
ExtendedMetaBox, AvifCondensedImageBoxTest,
Combine(/*width=*/Values(16), /*height=*/Values(24), /*depth=*/Values(8),
Values(AVIF_PIXEL_FORMAT_YUV444), Values(AVIF_PLANES_ALL),
Values(AVIF_RANGE_FULL), /*create_icc=*/Values(true),
/*create_exif=*/Values(true), /*create_xmp=*/Values(true),
/*create_clli=*/Values(true),
Values(AVIF_TRANSFORM_NONE,
AVIF_TRANSFORM_PASP | AVIF_TRANSFORM_CLAP |
AVIF_TRANSFORM_IROT | AVIF_TRANSFORM_IMIR)));
//------------------------------------------------------------------------------
} // namespace
} // namespace libavif