blob: 26c550ad2d98f153a5302882533cc8a3d7d8bcee [file] [log] [blame] [edit]
// Copyright 2017 PDFium 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 "core/fxge/dib/cstretchengine.h"
#include <utility>
#include "core/fpdfapi/page/cpdf_dib.h"
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fpdfapi/parser/cpdf_number.h"
#include "core/fpdfapi/parser/cpdf_stream.h"
#include "core/fxge/dib/fx_dib.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace {
// Discovered experimentally
constexpr uint32_t kTooBigSrcLen = 20;
constexpr uint32_t kTooBigDestLen = 32 * 1024 * 1024 + 1;
uint32_t PixelWeightSum(const CStretchEngine::PixelWeight* weights) {
uint32_t sum = 0;
for (int i = weights->m_SrcStart; i <= weights->m_SrcEnd; ++i) {
sum += weights->GetWeightForPosition(i);
}
return sum;
}
void ExecuteOneStretchTest(int32_t dest_width,
int32_t src_width,
const FXDIB_ResampleOptions& options) {
constexpr uint32_t kExpectedSum = CStretchEngine::kFixedPointOne;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(dest_width, 0, dest_width, src_width, 0,
src_width, options));
for (int32_t i = 0; i < dest_width; ++i) {
EXPECT_EQ(kExpectedSum, PixelWeightSum(table.GetPixelWeight(i)))
<< "for { " << src_width << ", " << dest_width << " } at " << i;
}
}
void ExecuteOneReversedStretchTest(int32_t dest_width,
int32_t src_width,
const FXDIB_ResampleOptions& options) {
constexpr uint32_t kExpectedSum = CStretchEngine::kFixedPointOne;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(-dest_width, 0, dest_width, src_width, 0,
src_width, options));
for (int32_t i = 0; i < dest_width; ++i) {
EXPECT_EQ(kExpectedSum, PixelWeightSum(table.GetPixelWeight(i)))
<< "for { " << src_width << ", " << dest_width << " } at " << i
<< " (reversed)";
}
}
void ExecuteStretchTests(const FXDIB_ResampleOptions& options) {
// Can't test everything, few random values chosen.
constexpr int32_t kDestWidths[] = {1, 2, 337, 512, 808, 2550};
constexpr int32_t kSrcWidths[] = {1, 2, 187, 256, 809, 1110};
for (int32_t src_width : kSrcWidths) {
for (int32_t dest_width : kDestWidths) {
ExecuteOneStretchTest(dest_width, src_width, options);
ExecuteOneReversedStretchTest(dest_width, src_width, options);
}
}
}
} // namespace
TEST(CStretchEngine, OverflowInCtor) {
FX_RECT clip_rect;
RetainPtr<CPDF_Dictionary> dict_obj = pdfium::MakeRetain<CPDF_Dictionary>();
dict_obj->SetNewFor<CPDF_Number>("Width", 71000);
dict_obj->SetNewFor<CPDF_Number>("Height", 12500);
RetainPtr<CPDF_Stream> stream =
pdfium::MakeRetain<CPDF_Stream>(nullptr, 0, std::move(dict_obj));
auto dib_source = pdfium::MakeRetain<CPDF_DIB>(nullptr, stream.Get());
dib_source->Load();
CStretchEngine engine(nullptr, FXDIB_Format::k8bppRgb, 500, 500, clip_rect,
dib_source, FXDIB_ResampleOptions());
EXPECT_TRUE(engine.GetResampleOptionsForTest().bInterpolateBilinear);
EXPECT_FALSE(engine.GetResampleOptionsForTest().bHalftone);
EXPECT_FALSE(engine.GetResampleOptionsForTest().bNoSmoothing);
EXPECT_FALSE(engine.GetResampleOptionsForTest().bLossy);
}
TEST(CStretchEngine, WeightRounding) {
FXDIB_ResampleOptions options;
ExecuteStretchTests(options);
}
TEST(CStretchEngine, WeightRoundingNoSmoothing) {
FXDIB_ResampleOptions options;
options.bNoSmoothing = true;
ExecuteStretchTests(options);
}
TEST(CStretchEngine, WeightRoundingBilinear) {
FXDIB_ResampleOptions options;
options.bInterpolateBilinear = true;
ExecuteStretchTests(options);
}
TEST(CStretchEngine, WeightRoundingNoSmoothingBilinear) {
FXDIB_ResampleOptions options;
options.bNoSmoothing = true;
options.bInterpolateBilinear = true;
ExecuteStretchTests(options);
}
TEST(CStretchEngine, ZeroLengthSrc) {
FXDIB_ResampleOptions options;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(100, 0, 100, 0, 0, 0, options));
}
TEST(CStretchEngine, ZeroLengthSrcNoSmoothing) {
FXDIB_ResampleOptions options;
options.bNoSmoothing = true;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(100, 0, 100, 0, 0, 0, options));
}
TEST(CStretchEngine, ZeroLengthSrcBilinear) {
FXDIB_ResampleOptions options;
options.bInterpolateBilinear = true;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(100, 0, 100, 0, 0, 0, options));
}
TEST(CStretchEngine, ZeroLengthSrcNoSmoothingBilinear) {
FXDIB_ResampleOptions options;
options.bNoSmoothing = true;
options.bInterpolateBilinear = true;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(100, 0, 100, 0, 0, 0, options));
}
TEST(CStretchEngine, ZeroLengthDest) {
FXDIB_ResampleOptions options;
CStretchEngine::WeightTable table;
ASSERT_TRUE(table.CalculateWeights(0, 0, 0, 100, 0, 100, options));
}
TEST(CStretchEngine, TooManyWeights) {
FXDIB_ResampleOptions options;
CStretchEngine::WeightTable table;
ASSERT_FALSE(table.CalculateWeights(kTooBigDestLen, 0, kTooBigDestLen,
kTooBigSrcLen, 0, kTooBigSrcLen,
options));
}