blob: ab7496c511566ab65e900f766d9f671c6666414f [file] [log] [blame]
// Copyright 2022 Google LLC
// SPDX-License-Identifier: BSD-2-Clause
#include "aviftest_helpers.h"
#include <algorithm>
#include <cassert>
#include <cmath>
#include <cstdint>
#include <cstdlib>
#include <cstring>
#include <string>
#include <vector>
#include "avif/avif.h"
#include "avif/internal.h"
#include "avifpng.h"
#include "avifutil.h"
namespace libavif {
namespace testutil {
//------------------------------------------------------------------------------
AvifRgbImage::AvifRgbImage(const avifImage* yuv, int rgbDepth,
avifRGBFormat rgbFormat) {
avifRGBImageSetDefaults(this, yuv);
depth = rgbDepth;
format = rgbFormat;
if (avifRGBImageAllocatePixels(this) != AVIF_RESULT_OK) {
std::abort();
}
}
AvifRwData::AvifRwData(AvifRwData&& other) : avifRWData{other} {
other.data = nullptr;
other.size = 0;
}
//------------------------------------------------------------------------------
RgbChannelOffsets GetRgbChannelOffsets(avifRGBFormat format) {
switch (format) {
case AVIF_RGB_FORMAT_RGB:
return {/*r=*/0, /*g=*/1, /*b=*/2, /*a=*/0};
case AVIF_RGB_FORMAT_RGBA:
return {/*r=*/0, /*g=*/1, /*b=*/2, /*a=*/3};
case AVIF_RGB_FORMAT_ARGB:
return {/*r=*/1, /*g=*/2, /*b=*/3, /*a=*/0};
case AVIF_RGB_FORMAT_BGR:
return {/*r=*/2, /*g=*/1, /*b=*/0, /*a=*/0};
case AVIF_RGB_FORMAT_BGRA:
return {/*r=*/2, /*g=*/1, /*b=*/0, /*a=*/3};
case AVIF_RGB_FORMAT_ABGR:
return {/*r=*/3, /*g=*/2, /*b=*/1, /*a=*/0};
case AVIF_RGB_FORMAT_RGB_565:
case AVIF_RGB_FORMAT_COUNT:
default:
return {/*r=*/0, /*g=*/0, /*b=*/0, /*a=*/0};
}
}
//------------------------------------------------------------------------------
AvifImagePtr CreateImage(int width, int height, int depth,
avifPixelFormat yuv_format, avifPlanesFlags planes,
avifRange yuv_range) {
AvifImagePtr image(avifImageCreate(width, height, depth, yuv_format),
avifImageDestroy);
if (!image) {
return {nullptr, nullptr};
}
image->yuvRange = yuv_range;
if (avifImageAllocatePlanes(image.get(), planes) != AVIF_RESULT_OK) {
return {nullptr, nullptr};
}
return image;
}
void FillImagePlain(avifImage* image, const uint32_t yuva[4]) {
for (avifChannelIndex c :
{AVIF_CHAN_Y, AVIF_CHAN_U, AVIF_CHAN_V, AVIF_CHAN_A}) {
const uint32_t plane_width = avifImagePlaneWidth(image, c);
// 0 for A if no alpha and 0 for UV if 4:0:0.
const uint32_t plane_height = avifImagePlaneHeight(image, c);
uint8_t* row = avifImagePlane(image, c);
const uint32_t row_bytes = avifImagePlaneRowBytes(image, c);
for (uint32_t y = 0; y < plane_height; ++y) {
if (avifImageUsesU16(image)) {
std::fill(reinterpret_cast<uint16_t*>(row),
reinterpret_cast<uint16_t*>(row) + plane_width,
static_cast<uint16_t>(yuva[c]));
} else {
std::fill(row, row + plane_width, static_cast<uint8_t>(yuva[c]));
}
row += row_bytes;
}
}
}
void FillImageGradient(avifImage* image) {
for (avifChannelIndex c :
{AVIF_CHAN_Y, AVIF_CHAN_U, AVIF_CHAN_V, AVIF_CHAN_A}) {
const uint32_t limitedRangeMin =
c == AVIF_CHAN_Y ? 16 << (image->depth - 8) : 0;
const uint32_t limitedRangeMax = (c == AVIF_CHAN_Y ? 219 : 224)
<< (image->depth - 8);
const uint32_t plane_width = avifImagePlaneWidth(image, c);
// 0 for A if no alpha and 0 for UV if 4:0:0.
const uint32_t plane_height = avifImagePlaneHeight(image, c);
uint8_t* row = avifImagePlane(image, c);
const uint32_t row_bytes = avifImagePlaneRowBytes(image, c);
for (uint32_t y = 0; y < plane_height; ++y) {
for (uint32_t x = 0; x < plane_width; ++x) {
uint32_t value;
if (image->yuvRange == AVIF_RANGE_FULL || c == AVIF_CHAN_A) {
value = (x + y) * ((1u << image->depth) - 1u) /
std::max(1u, plane_width + plane_height - 2);
} else {
value = limitedRangeMin +
(x + y) * (limitedRangeMax - limitedRangeMin) /
std::max(1u, plane_width + plane_height - 2);
}
if (avifImageUsesU16(image)) {
reinterpret_cast<uint16_t*>(row)[x] = static_cast<uint16_t>(value);
} else {
row[x] = static_cast<uint8_t>(value);
}
}
row += row_bytes;
}
}
}
namespace {
template <typename PixelType>
void FillImageChannel(avifRGBImage* image, uint32_t channel_offset,
uint32_t value) {
const uint32_t channel_count = avifRGBFormatChannelCount(image->format);
assert(channel_offset < channel_count);
for (uint32_t y = 0; y < image->height; ++y) {
PixelType* pixel =
reinterpret_cast<PixelType*>(image->pixels + image->rowBytes * y);
for (uint32_t x = 0; x < image->width; ++x) {
pixel[channel_offset] = static_cast<PixelType>(value);
pixel += channel_count;
}
}
}
} // namespace
void FillImageChannel(avifRGBImage* image, uint32_t channel_offset,
uint32_t value) {
(image->depth <= 8)
? FillImageChannel<uint8_t>(image, channel_offset, value)
: FillImageChannel<uint16_t>(image, channel_offset, value);
}
//------------------------------------------------------------------------------
bool AreByteSequencesEqual(const uint8_t data1[], size_t data1_length,
const uint8_t data2[], size_t data2_length) {
if (data1_length != data2_length) return false;
return data1_length == 0 || std::equal(data1, data1 + data1_length, data2);
}
bool AreByteSequencesEqual(const avifRWData& data1, const avifRWData& data2) {
return AreByteSequencesEqual(data1.data, data1.size, data2.data, data2.size);
}
// Returns true if image1 and image2 are identical.
bool AreImagesEqual(const avifImage& image1, const avifImage& image2,
bool ignore_alpha) {
if (image1.width != image2.width || image1.height != image2.height ||
image1.depth != image2.depth || image1.yuvFormat != image2.yuvFormat ||
image1.yuvRange != image2.yuvRange) {
return false;
}
assert(image1.width * image1.height > 0);
if (image1.clli.maxCLL != image2.clli.maxCLL ||
image1.clli.maxPALL != image2.clli.maxPALL) {
return false;
}
if (image1.transformFlags != image2.transformFlags ||
((image1.transformFlags & AVIF_TRANSFORM_PASP) &&
std::memcmp(&image1.pasp, &image2.pasp, sizeof(image1.pasp))) ||
((image1.transformFlags & AVIF_TRANSFORM_CLAP) &&
std::memcmp(&image1.clap, &image2.clap, sizeof(image1.clap))) ||
((image1.transformFlags & AVIF_TRANSFORM_IROT) &&
std::memcmp(&image1.irot, &image2.irot, sizeof(image1.irot))) ||
((image1.transformFlags & AVIF_TRANSFORM_IMIR) &&
std::memcmp(&image1.imir, &image2.imir, sizeof(image1.imir)))) {
return false;
}
for (avifChannelIndex c :
{AVIF_CHAN_Y, AVIF_CHAN_U, AVIF_CHAN_V, AVIF_CHAN_A}) {
if (ignore_alpha && c == AVIF_CHAN_A) continue;
const uint8_t* row1 = avifImagePlane(&image1, c);
const uint8_t* row2 = avifImagePlane(&image2, c);
if (!row1 != !row2) {
// Maybe one image contains an opaque alpha channel while the other has no
// alpha channel, but they should still be considered equal.
if (c == AVIF_CHAN_A && avifImageIsOpaque(&image1) &&
avifImageIsOpaque(&image2)) {
continue;
}
return false;
}
const uint32_t row_bytes1 = avifImagePlaneRowBytes(&image1, c);
const uint32_t row_bytes2 = avifImagePlaneRowBytes(&image2, c);
const uint32_t plane_width = avifImagePlaneWidth(&image1, c);
// 0 for A if no alpha and 0 for UV if 4:0:0.
const uint32_t plane_height = avifImagePlaneHeight(&image1, c);
for (uint32_t y = 0; y < plane_height; ++y) {
if (avifImageUsesU16(&image1)) {
if (!std::equal(reinterpret_cast<const uint16_t*>(row1),
reinterpret_cast<const uint16_t*>(row1) + plane_width,
reinterpret_cast<const uint16_t*>(row2))) {
return false;
}
} else {
if (!std::equal(row1, row1 + plane_width, row2)) {
return false;
}
}
row1 += row_bytes1;
row2 += row_bytes2;
}
}
return AreByteSequencesEqual(image1.icc, image2.icc) &&
AreByteSequencesEqual(image1.exif, image2.exif) &&
AreByteSequencesEqual(image1.xmp, image2.xmp);
}
namespace {
template <typename Sample>
uint64_t SquaredDiffSum(const Sample* samples1, const Sample* samples2,
uint32_t num_samples) {
uint64_t sum = 0;
for (uint32_t i = 0; i < num_samples; ++i) {
const int32_t diff = static_cast<int32_t>(samples1[i]) - samples2[i];
sum += diff * diff;
}
return sum;
}
} // namespace
double GetPsnr(const avifImage& image1, const avifImage& image2,
bool ignore_alpha) {
if (image1.width != image2.width || image1.height != image2.height ||
image1.depth != image2.depth || image1.yuvFormat != image2.yuvFormat ||
image1.yuvRange != image2.yuvRange) {
return -1;
}
assert(image1.width * image1.height > 0);
uint64_t squared_diff_sum = 0;
uint32_t num_samples = 0;
const uint32_t max_sample_value = (1 << image1.depth) - 1;
for (avifChannelIndex c :
{AVIF_CHAN_Y, AVIF_CHAN_U, AVIF_CHAN_V, AVIF_CHAN_A}) {
if (ignore_alpha && c == AVIF_CHAN_A) continue;
const uint32_t plane_width = std::max(avifImagePlaneWidth(&image1, c),
avifImagePlaneWidth(&image2, c));
const uint32_t plane_height = std::max(avifImagePlaneHeight(&image1, c),
avifImagePlaneHeight(&image2, c));
if (plane_width == 0 || plane_height == 0) continue;
const uint8_t* row1 = avifImagePlane(&image1, c);
const uint8_t* row2 = avifImagePlane(&image2, c);
if (!row1 != !row2 && c != AVIF_CHAN_A) {
return -1;
}
uint32_t row_bytes1 = avifImagePlaneRowBytes(&image1, c);
uint32_t row_bytes2 = avifImagePlaneRowBytes(&image2, c);
// Consider missing alpha planes as samples set to the maximum value.
std::vector<uint8_t> opaque_alpha_samples;
if (!row1 != !row2) {
opaque_alpha_samples.resize(std::max(row_bytes1, row_bytes2));
if (avifImageUsesU16(&image1)) {
uint16_t* opaque_alpha_samples_16b =
reinterpret_cast<uint16_t*>(opaque_alpha_samples.data());
std::fill(opaque_alpha_samples_16b,
opaque_alpha_samples_16b + plane_width,
static_cast<int16_t>(max_sample_value));
} else {
std::fill(opaque_alpha_samples.begin(), opaque_alpha_samples.end(),
uint8_t{255});
}
if (!row1) {
row1 = opaque_alpha_samples.data();
row_bytes1 = 0;
} else {
row2 = opaque_alpha_samples.data();
row_bytes2 = 0;
}
}
for (uint32_t y = 0; y < plane_height; ++y) {
if (avifImageUsesU16(&image1)) {
squared_diff_sum += SquaredDiffSum(
reinterpret_cast<const uint16_t*>(row1),
reinterpret_cast<const uint16_t*>(row2), plane_width);
} else {
squared_diff_sum += SquaredDiffSum(row1, row2, plane_width);
}
row1 += row_bytes1;
row2 += row_bytes2;
num_samples += plane_width;
}
}
if (squared_diff_sum == 0) {
return 99.0;
}
const double normalized_error =
squared_diff_sum /
(static_cast<double>(num_samples) * max_sample_value * max_sample_value);
if (normalized_error <= std::numeric_limits<double>::epsilon()) {
return 98.99; // Very small distortion but not lossless.
}
return std::min(-10 * std::log10(normalized_error), 98.99);
}
bool AreImagesEqual(const avifRGBImage& image1, const avifRGBImage& image2) {
if (image1.width != image2.width || image1.height != image2.height ||
image1.depth != image2.depth || image1.format != image2.format ||
image1.alphaPremultiplied != image2.alphaPremultiplied ||
image1.isFloat != image2.isFloat) {
return false;
}
const uint8_t* row1 = image1.pixels;
const uint8_t* row2 = image2.pixels;
const unsigned int row_width = image1.width * avifRGBImagePixelSize(&image1);
for (unsigned int y = 0; y < image1.height; ++y) {
if (!std::equal(row1, row1 + row_width, row2)) {
return false;
}
row1 += image1.rowBytes;
row2 += image2.rowBytes;
}
return true;
}
avifResult MergeGrid(int grid_cols, int grid_rows,
const std::vector<AvifImagePtr>& cells,
avifImage* merged) {
std::vector<const avifImage*> ptrs(cells.size());
for (size_t i = 0; i < cells.size(); ++i) {
ptrs[i] = cells[i].get();
}
return MergeGrid(grid_cols, grid_rows, ptrs, merged);
}
avifResult MergeGrid(int grid_cols, int grid_rows,
const std::vector<const avifImage*>& cells,
avifImage* merged) {
const uint32_t tile_width = cells[0]->width;
const uint32_t tile_height = cells[0]->height;
const uint32_t grid_width =
(grid_cols - 1) * tile_width + cells.back()->width;
const uint32_t grid_height =
(grid_rows - 1) * tile_height + cells.back()->height;
testutil::AvifImagePtr view(avifImageCreateEmpty(), avifImageDestroy);
AVIF_CHECKERR(view, AVIF_RESULT_OUT_OF_MEMORY);
avifCropRect rect = {};
for (int j = 0; j < grid_rows; ++j) {
rect.x = 0;
for (int i = 0; i < grid_cols; ++i) {
const avifImage* image = cells[j * grid_cols + i];
rect.width = image->width;
rect.height = image->height;
AVIF_CHECKRES(avifImageSetViewRect(view.get(), merged, &rect));
avifImageCopySamples(/*dstImage=*/view.get(), image, AVIF_PLANES_ALL);
assert(!view->imageOwnsYUVPlanes);
rect.x += rect.width;
}
rect.y += rect.height;
}
if ((rect.x != grid_width) || (rect.y != grid_height)) {
return AVIF_RESULT_UNKNOWN_ERROR;
}
return AVIF_RESULT_OK;
}
//------------------------------------------------------------------------------
AvifImagePtr ReadImage(const char* folder_path, const char* file_name,
avifPixelFormat requested_format, int requested_depth,
avifChromaDownsampling chroma_downsampling,
avifBool ignore_icc, avifBool ignore_exif,
avifBool ignore_xmp, avifBool allow_changing_cicp,
avifBool ignore_gain_map) {
testutil::AvifImagePtr image(avifImageCreateEmpty(), avifImageDestroy);
if (!image ||
avifReadImage((std::string(folder_path) + file_name).c_str(),
requested_format, requested_depth, chroma_downsampling,
ignore_icc, ignore_exif, ignore_xmp, allow_changing_cicp,
ignore_gain_map, image.get(),
/*outDepth=*/nullptr, /*sourceTiming=*/nullptr,
/*frameIter=*/nullptr) == AVIF_APP_FILE_FORMAT_UNKNOWN) {
return {nullptr, nullptr};
}
return image;
}
bool WriteImage(const avifImage* image, const char* file_path) {
if (!image || !file_path) return false;
const size_t str_len = std::strlen(file_path);
if (str_len >= 4 && !std::strncmp(file_path + str_len - 4, ".png", 4)) {
return avifPNGWrite(file_path, image, /*requestedDepth=*/0,
AVIF_CHROMA_UPSAMPLING_BEST_QUALITY,
/*compressionLevel=*/0);
}
// Other formats are not supported.
return false;
}
AvifRwData Encode(const avifImage* image, int speed) {
testutil::AvifEncoderPtr encoder(avifEncoderCreate(), avifEncoderDestroy);
if (!encoder) return {};
encoder->speed = speed;
testutil::AvifRwData bytes;
if (avifEncoderWrite(encoder.get(), image, &bytes) != AVIF_RESULT_OK) {
return {};
}
return bytes;
}
AvifImagePtr Decode(const uint8_t* bytes, size_t num_bytes) {
testutil::AvifImagePtr decoded(avifImageCreateEmpty(), avifImageDestroy);
testutil::AvifDecoderPtr decoder(avifDecoderCreate(), avifDecoderDestroy);
if (!decoded || !decoder ||
(avifDecoderReadMemory(decoder.get(), decoded.get(), bytes, num_bytes) !=
AVIF_RESULT_OK)) {
return {nullptr, nullptr};
}
return decoded;
}
bool Av1EncoderAvailable() {
const char* encoding_codec =
avifCodecName(AVIF_CODEC_CHOICE_AUTO, AVIF_CODEC_FLAG_CAN_ENCODE);
return encoding_codec != nullptr && std::string(encoding_codec) != "avm";
}
bool Av1DecoderAvailable() {
const char* decoding_codec =
avifCodecName(AVIF_CODEC_CHOICE_AUTO, AVIF_CODEC_FLAG_CAN_DECODE);
return decoding_codec != nullptr && std::string(decoding_codec) != "avm";
}
//------------------------------------------------------------------------------
static avifResult avifIOLimitedReaderRead(avifIO* io, uint32_t readFlags,
uint64_t offset, size_t size,
avifROData* out) {
auto reader = reinterpret_cast<AvifIOLimitedReader*>(io);
if (offset > UINT64_MAX - size) {
return AVIF_RESULT_IO_ERROR;
}
if (offset + size > reader->clamp) {
return AVIF_RESULT_WAITING_ON_IO;
}
return reader->underlyingIO->read(reader->underlyingIO, readFlags, offset,
size, out);
}
static void avifIOLimitedReaderDestroy(avifIO* io) {
auto reader = reinterpret_cast<AvifIOLimitedReader*>(io);
reader->underlyingIO->destroy(reader->underlyingIO);
delete reader;
}
avifIO* AvifIOCreateLimitedReader(avifIO* underlyingIO, uint64_t clamp) {
return reinterpret_cast<avifIO*>(
new AvifIOLimitedReader{{
avifIOLimitedReaderDestroy,
avifIOLimitedReaderRead,
nullptr,
underlyingIO->sizeHint,
underlyingIO->persistent,
nullptr,
},
underlyingIO,
clamp});
}
//------------------------------------------------------------------------------
} // namespace testutil
} // namespace libavif