blob: 428c896361abf46cda1f15ae211532d56b1ff767 [file] [log] [blame]
/*
* Copyright 2023 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkBitmap.h"
#include "include/core/SkBlendMode.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkClipOp.h"
#include "include/core/SkColor.h"
#include "include/core/SkColorFilter.h"
#include "include/core/SkColorSpace.h"
#include "include/core/SkColorType.h"
#include "include/core/SkData.h"
#include "include/core/SkImage.h"
#include "include/core/SkImageInfo.h"
#include "include/core/SkMatrix.h"
#include "include/core/SkPaint.h"
#include "include/core/SkPoint.h"
#include "include/core/SkRect.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkSamplingOptions.h"
#include "include/core/SkScalar.h"
#include "include/core/SkSize.h"
#include "include/core/SkString.h"
#include "include/core/SkTileMode.h"
#include "include/private/SkColorData.h"
#include "include/private/base/SkAssert.h"
#include "include/private/base/SkDebug.h"
#include "include/private/base/SkTArray.h"
#include "include/private/base/SkTo.h"
#include "src/core/SkDevice.h"
#include "src/core/SkImageFilterTypes.h"
#include "src/core/SkMatrixPriv.h"
#include "src/core/SkRectPriv.h"
#include "src/core/SkSpecialImage.h"
#include "src/effects/colorfilters/SkColorFilterBase.h"
#include "src/gpu/ganesh/image/GrImageUtils.h"
#include "tests/CtsEnforcement.h"
#include "tests/Test.h"
#include "tools/EncodeUtils.h"
#include "tools/gpu/ContextType.h"
#include <cmath>
#include <initializer_list>
#include <optional>
#include <string>
#include <utility>
#include <variant>
#include <vector>
#if defined(SK_GRAPHITE)
#include "include/gpu/graphite/Context.h"
#include "src/gpu/graphite/ContextPriv.h"
#include "src/gpu/graphite/RecorderPriv.h"
#include "src/gpu/graphite/SpecialImage_Graphite.h"
#include "src/gpu/graphite/TextureProxyView.h"
#include "src/gpu/graphite/TextureUtils.h"
#endif
#if defined(SK_GANESH)
#include "include/gpu/GrDirectContext.h"
#include "include/gpu/GrRecordingContext.h"
#include "include/gpu/GrTypes.h"
struct GrContextOptions;
#endif
class SkShader;
using namespace skia_private;
using namespace skif;
// NOTE: Not in anonymous so that FilterResult can friend it
class FilterResultTestAccess {
using BoundsAnalysis = FilterResult::BoundsAnalysis;
public:
static void Draw(const skif::Context& ctx,
SkDevice* device,
const skif::FilterResult& image,
bool preserveDeviceState) {
image.draw(ctx, device, preserveDeviceState, /*blender=*/nullptr);
}
static sk_sp<SkShader> AsShader(const skif::Context& ctx,
const skif::FilterResult& image,
const skif::LayerSpace<SkIRect>& sampleBounds) {
return image.asShader(ctx, FilterResult::kDefaultSampling,
FilterResult::ShaderFlags::kNone, sampleBounds);
}
static sk_sp<SkShader> StrictShader(const skif::Context& ctx,
const skif::FilterResult& image) {
auto analysis = image.analyzeBounds(ctx.desiredOutput());
if (analysis & FilterResult::BoundsAnalysis::kRequiresLayerCrop) {
// getAnalyzedShaderView() doesn't include the layer crop, this will be handled by
// the FilterResultImageResolver.
return nullptr;
} else {
// Add flags to ensure no deferred effects or clamping logic are optimized away.
analysis |= BoundsAnalysis::kDstBoundsNotCovered;
analysis |= BoundsAnalysis::kRequiresShaderTiling;
if (image.tileMode() == SkTileMode::kDecal) {
analysis |= BoundsAnalysis::kRequiresDecalInLayerSpace;
}
return image.getAnalyzedShaderView(ctx, image.sampling(), analysis);
}
}
static skif::FilterResult Rescale(const skif::Context& ctx,
const skif::FilterResult& image,
const skif::LayerSpace<SkSize> scale) {
return image.rescale(ctx, scale, /*enforceDecal=*/false);
}
static void TrackStats(skif::Context* ctx, skif::Stats* stats) {
ctx->fStats = stats;
}
static bool IsIntegerTransform(const skif::FilterResult& image) {
SkMatrix m = SkMatrix(image.fTransform);
return m.isTranslate() &&
SkScalarIsInt(m.getTranslateX()) &&
SkScalarIsInt(m.getTranslateY());
}
static bool IsShaderTilingExpected(const skif::Context& ctx,
const skif::FilterResult& image,
bool rescaling) {
if (image.tileMode() == SkTileMode::kClamp) {
return false;
}
if (image.tileMode() == SkTileMode::kDecal &&
image.fBoundary == FilterResult::PixelBoundary::kTransparent) {
return false;
}
auto analysis = image.analyzeBounds(ctx.desiredOutput());
if (!(analysis & BoundsAnalysis::kHasLayerFillingEffect) &&
(image.tileMode() == SkTileMode::kRepeat || image.tileMode() == SkTileMode::kMirror ||
(image.tileMode() == SkTileMode::kDecal && !rescaling))) {
return false;
}
// If we got here, it's either a mirror/repeat tile mode that's visible so a shader has to
// be used if the image isn't HW tileable; OR it's a decal tile mode without transparent
// padding that can't be drawn directly (in this case hasLayerFillingEffect implies a
// color filter that has to evaluate the decal'ed sampling).
// TODO(b/323886180): Rescaling with decal images does not draw directly but should, so it
// will eventually avoid the expensive decal shader.
return true;
}
static bool IsShaderClampingExpected(const skif::Context& ctx,
const skif::FilterResult& image,
bool rescaling) {
auto analysis = image.analyzeBounds(ctx.desiredOutput());
if (analysis & BoundsAnalysis::kHasLayerFillingEffect ||
(image.tileMode() == SkTileMode::kDecal && rescaling)) {
// The image won't be drawn directly so some form of shader is needed. The faster clamp
// can be used when clamping explicitly or decal-with-transparent-padding.
// TODO(b/323886180): Once rescaling can draw decals directly, decal-with-transparent
// padding should only need clamping when there's a layer-filling color filter as well.
if (image.tileMode() == SkTileMode::kClamp ||
(image.tileMode() == SkTileMode::kDecal &&
image.fBoundary == FilterResult::PixelBoundary::kTransparent)) {
return true;
} else {
// These cases should be covered by the more expensive shader tiling, but if we
// are rescaling with a deferrable tile mode, it can still be converted to a clamp.
SkASSERT(IsShaderTilingExpected(ctx, image, rescaling));
return rescaling;
}
}
// If we got here, it will be drawn directly but a clamp can be needed if the data outside
// the image is unknown and sampling might pull those values in accidentally.
return image.fBoundary == FilterResult::PixelBoundary::kUnknown;
}
};
namespace {
// Parameters controlling the fuzziness matching of expected and actual images.
// NOTE: When image fuzzy diffing fails it will print the expected image, the actual image, and
// an "error" image where all bad pixels have been set to red. You can select all three base64
// encoded PNGs, copy them, and run the following command to view in detail:
// xsel -o | viewer --file stdin
static constexpr float kRGBTolerance = 8.f / 255.f;
static constexpr float kAATolerance = 2.f / 255.f;
static constexpr float kDefaultMaxAllowedPercentImageDiff = 1.f;
static const float kFuzzyKernel[3][3] = {{0.9f, 0.9f, 0.9f},
{0.9f, 1.0f, 0.9f},
{0.9f, 0.9f, 0.9f}};
static_assert(std::size(kFuzzyKernel) == std::size(kFuzzyKernel[0]), "Kernel must be square");
static constexpr int kKernelSize = std::size(kFuzzyKernel);
static constexpr bool kLogAllBitmaps = false; // Spammy, recommend limiting test cases being run
bool colorfilter_equals(const SkColorFilter* actual, const SkColorFilter* expected) {
if (!actual || !expected) {
return !actual && !expected; // both null
}
// The two filter objects are equal if they serialize to the same structure
sk_sp<SkData> actualData = actual->serialize();
sk_sp<SkData> expectedData = expected->serialize();
return actualData && actualData->equals(expectedData.get());
}
void clear_device(SkDevice* device) {
SkPaint p;
p.setColor4f(SkColors::kTransparent, /*colorSpace=*/nullptr);
p.setBlendMode(SkBlendMode::kSrc);
device->drawPaint(p);
}
static constexpr SkTileMode kTileModes[4] = {SkTileMode::kClamp,
SkTileMode::kRepeat,
SkTileMode::kMirror,
SkTileMode::kDecal};
enum class Expect {
kDeferredImage, // i.e. modified properties of FilterResult instead of rendering
kNewImage, // i.e. rendered a new image before modifying other properties
kEmptyImage, // i.e. everything is transparent black
};
class ApplyAction {
struct TransformParams {
LayerSpace<SkMatrix> fMatrix;
SkSamplingOptions fSampling;
};
struct CropParams {
LayerSpace<SkIRect> fRect;
SkTileMode fTileMode;
// Sometimes the expected bounds due to cropping and tiling are too hard to automate with
// simple test code.
std::optional<LayerSpace<SkIRect>> fExpectedBounds;
};
struct RescaleParams {
LayerSpace<SkSize> fScale;
};
public:
ApplyAction(const SkMatrix& transform,
const SkSamplingOptions& sampling,
Expect expectation,
const SkSamplingOptions& expectedSampling,
SkTileMode expectedTileMode,
sk_sp<SkColorFilter> expectedColorFilter)
: fAction{TransformParams{LayerSpace<SkMatrix>(transform), sampling}}
, fExpectation(expectation)
, fExpectedSampling(expectedSampling)
, fExpectedTileMode(expectedTileMode)
, fExpectedColorFilter(std::move(expectedColorFilter)) {}
ApplyAction(const SkIRect& cropRect,
SkTileMode tileMode,
std::optional<LayerSpace<SkIRect>> expectedBounds,
Expect expectation,
const SkSamplingOptions& expectedSampling,
SkTileMode expectedTileMode,
sk_sp<SkColorFilter> expectedColorFilter)
: fAction{CropParams{LayerSpace<SkIRect>(cropRect), tileMode, expectedBounds}}
, fExpectation(expectation)
, fExpectedSampling(expectedSampling)
, fExpectedTileMode(expectedTileMode)
, fExpectedColorFilter(std::move(expectedColorFilter)) {}
ApplyAction(sk_sp<SkColorFilter> colorFilter,
Expect expectation,
const SkSamplingOptions& expectedSampling,
SkTileMode expectedTileMode,
sk_sp<SkColorFilter> expectedColorFilter)
: fAction(std::move(colorFilter))
, fExpectation(expectation)
, fExpectedSampling(expectedSampling)
, fExpectedTileMode(expectedTileMode)
, fExpectedColorFilter(std::move(expectedColorFilter)) {}
ApplyAction(LayerSpace<SkSize> scale,
Expect expectation,
const SkSamplingOptions& expectedSampling,
SkTileMode expectedTileMode,
sk_sp<SkColorFilter> expectedColorFilter)
: fAction(RescaleParams{scale})
, fExpectation(expectation)
, fExpectedSampling(expectedSampling)
, fExpectedTileMode(expectedTileMode)
, fExpectedColorFilter(std::move(expectedColorFilter)) {}
// Test-simplified logic for bounds propagation similar to how image filters calculate bounds
// while evaluating a filter DAG, which is outside of skif::FilterResult's responsibilities.
LayerSpace<SkIRect> requiredInput(const LayerSpace<SkIRect>& desiredOutput) const {
if (auto* t = std::get_if<TransformParams>(&fAction)) {
LayerSpace<SkIRect> out;
return t->fMatrix.inverseMapRect(desiredOutput, &out)
? out : LayerSpace<SkIRect>::Empty();
} else if (auto* c = std::get_if<CropParams>(&fAction)) {
LayerSpace<SkIRect> intersection = c->fRect;
if (c->fTileMode == SkTileMode::kDecal && !intersection.intersect(desiredOutput)) {
intersection = LayerSpace<SkIRect>::Empty();
}
return intersection;
} else if (std::holds_alternative<sk_sp<SkColorFilter>>(fAction) ||
std::holds_alternative<RescaleParams>(fAction)) {
return desiredOutput;
}
SkUNREACHABLE;
}
// Performs the action to be tested
FilterResult apply(const Context& ctx, const FilterResult& in) const {
if (auto* t = std::get_if<TransformParams>(&fAction)) {
return in.applyTransform(ctx, t->fMatrix, t->fSampling);
} else if (auto* c = std::get_if<CropParams>(&fAction)) {
return in.applyCrop(ctx, c->fRect, c->fTileMode);
} else if (auto* cf = std::get_if<sk_sp<SkColorFilter>>(&fAction)) {
return in.applyColorFilter(ctx, *cf);
} else if (auto* s = std::get_if<RescaleParams>(&fAction)) {
return FilterResultTestAccess::Rescale(ctx, in, s->fScale);
}
SkUNREACHABLE;
}
Expect expectation() const { return fExpectation; }
const SkSamplingOptions& expectedSampling() const { return fExpectedSampling; }
SkTileMode expectedTileMode() const { return fExpectedTileMode; }
const SkColorFilter* expectedColorFilter() const { return fExpectedColorFilter.get(); }
bool isRescaling() const { return std::holds_alternative<RescaleParams>(fAction); }
int expectedOffscreenSurfaces() const {
if (fExpectation != Expect::kNewImage) {
return 0;
}
if (auto* s = std::get_if<RescaleParams>(&fAction)) {
float minScale = std::min(s->fScale.width(), s->fScale.height());
if (minScale >= 1.f - 0.001f) {
return 1;
} else {
int steps = 0;
do {
steps++;
minScale *= 2.f;
} while(minScale < 0.8f);
return steps;
}
} else {
return 1;
}
}
LayerSpace<SkIRect> expectedBounds(const LayerSpace<SkIRect>& inputBounds) const {
// This assumes anything outside 'inputBounds' is transparent black.
if (auto* t = std::get_if<TransformParams>(&fAction)) {
if (inputBounds.isEmpty()) {
return LayerSpace<SkIRect>::Empty();
}
return t->fMatrix.mapRect(inputBounds);
} else if (auto* c = std::get_if<CropParams>(&fAction)) {
if (c->fExpectedBounds) {
return *c->fExpectedBounds;
}
LayerSpace<SkIRect> intersection = c->fRect;
if (!intersection.intersect(inputBounds)) {
return LayerSpace<SkIRect>::Empty();
}
return c->fTileMode == SkTileMode::kDecal
? intersection : LayerSpace<SkIRect>(SkRectPriv::MakeILarge());
} else if (auto* cf = std::get_if<sk_sp<SkColorFilter>>(&fAction)) {
if (as_CFB(*cf)->affectsTransparentBlack()) {
// Fills out infinitely
return LayerSpace<SkIRect>(SkRectPriv::MakeILarge());
} else {
return inputBounds;
}
} else if (std::holds_alternative<RescaleParams>(fAction)) {
return inputBounds;
}
SkUNREACHABLE;
}
sk_sp<SkSpecialImage> renderExpectedImage(const Context& ctx,
sk_sp<SkSpecialImage> source,
LayerSpace<SkIPoint> origin,
const LayerSpace<SkIRect>& desiredOutput) const {
SkASSERT(source);
Expect effectiveExpectation = fExpectation;
SkISize size(desiredOutput.size());
if (desiredOutput.isEmpty()) {
size = {1, 1};
effectiveExpectation = Expect::kEmptyImage;
}
auto device = ctx.backend()->makeDevice(size, ctx.refColorSpace());
SkCanvas canvas{device};
canvas.clear(SK_ColorTRANSPARENT);
canvas.translate(-desiredOutput.left(), -desiredOutput.top());
LayerSpace<SkIRect> sourceBounds{
SkIRect::MakeXYWH(origin.x(), origin.y(), source->width(), source->height())};
LayerSpace<SkIRect> expectedBounds = this->expectedBounds(sourceBounds);
canvas.clipIRect(SkIRect(expectedBounds), SkClipOp::kIntersect);
if (effectiveExpectation != Expect::kEmptyImage) {
SkPaint paint;
paint.setAntiAlias(true);
paint.setBlendMode(SkBlendMode::kSrc);
// Start with NN to match exact subsetting FilterResult does for deferred images
SkSamplingOptions sampling = {};
SkTileMode tileMode = SkTileMode::kDecal;
if (auto* t = std::get_if<TransformParams>(&fAction)) {
SkMatrix m{t->fMatrix};
// FilterResult treats default/bilerp filtering as NN when it has an integer
// translation, so only change 'sampling' when that is not the case.
if (!m.isTranslate() ||
!SkScalarIsInt(m.getTranslateX()) ||
!SkScalarIsInt(m.getTranslateY())) {
sampling = t->fSampling;
}
canvas.concat(m);
} else if (auto* c = std::get_if<CropParams>(&fAction)) {
LayerSpace<SkIRect> imageBounds(
SkIRect::MakeXYWH(origin.x(), origin.y(),
source->width(), source->height()));
if (c->fTileMode == SkTileMode::kDecal || imageBounds.contains(c->fRect)) {
// Extract a subset of the image
SkAssertResult(imageBounds.intersect(c->fRect));
source = source->makeSubset({imageBounds.left() - origin.x(),
imageBounds.top() - origin.y(),
imageBounds.right() - origin.x(),
imageBounds.bottom() - origin.y()});
origin = imageBounds.topLeft();
} else {
// A non-decal tile mode where the image doesn't cover the crop requires the
// image to be padded out with transparency so the tiling matches 'fRect'.
SkISize paddedSize = SkISize(c->fRect.size());
auto paddedDevice = ctx.backend()->makeDevice(paddedSize, ctx.refColorSpace());
clear_device(paddedDevice.get());
paddedDevice->drawSpecial(source.get(),
SkMatrix::Translate(origin.x() - c->fRect.left(),
origin.y() - c->fRect.top()),
/*sampling=*/{},
/*paint=*/{});
source = paddedDevice->snapSpecial(SkIRect::MakeSize(paddedSize));
origin = c->fRect.topLeft();
}
tileMode = c->fTileMode;
} else if (auto* cf = std::get_if<sk_sp<SkColorFilter>>(&fAction)) {
paint.setColorFilter(*cf);
} else if (auto* s = std::get_if<RescaleParams>(&fAction)) {
// Don't redraw with an identity scale since sampling errors creep in on some GPUs
if (s->fScale.width() != 1.f || s->fScale.height() != 1.f) {
SkISize lowResSize = {sk_float_ceil2int(source->width() * s->fScale.width()),
sk_float_ceil2int(source->height() * s->fScale.height())};
while (source->width() != lowResSize.width() ||
source->height() != lowResSize.height()) {
float sx = std::max(0.5f, lowResSize.width() / (float) source->width());
float sy = std::max(0.5f, lowResSize.height() / (float) source->height());
SkISize stepSize = {sk_float_ceil2int(source->width() * sx),
sk_float_ceil2int(source->height() * sy)};
auto stepDevice = ctx.backend()->makeDevice(stepSize, ctx.refColorSpace());
clear_device(stepDevice.get());
stepDevice->drawSpecial(source.get(),
SkMatrix::Scale(sx, sy),
SkFilterMode::kLinear,
/*paint=*/{});
source = stepDevice->snapSpecial(SkIRect::MakeSize(stepSize));
}
// Adjust to draw the low-res image upscaled to fill the original image bounds
sampling = SkFilterMode::kLinear;
tileMode = SkTileMode::kClamp;
canvas.translate(origin.x(), origin.y());
canvas.scale(1.f / s->fScale.width(), 1.f / s->fScale.height());
origin = LayerSpace<SkIPoint>({0, 0});
}
}
// else it's a rescale action, but for the expected image leave it unmodified.
paint.setShader(source->asShader(tileMode,
sampling,
SkMatrix::Translate(origin.x(), origin.y())));
canvas.drawPaint(paint);
}
return device->snapSpecial(SkIRect::MakeSize(size));
}
private:
// Action
std::variant<TransformParams, // for applyTransform()
CropParams, // for applyCrop()
sk_sp<SkColorFilter>,// for applyColorFilter()
RescaleParams // for rescale()
> fAction;
// Expectation
Expect fExpectation;
SkSamplingOptions fExpectedSampling;
SkTileMode fExpectedTileMode;
sk_sp<SkColorFilter> fExpectedColorFilter;
// The expected desired outputs and layer bounds are calculated automatically based on the
// action type and parameters to simplify test case specification.
};
class FilterResultImageResolver {
public:
enum class Method {
kImageAndOffset,
kDrawToCanvas,
kShader,
kClippedShader,
kStrictShader // Only used to check image correctness when stats reported an optimization
};
FilterResultImageResolver(Method method) : fMethod(method) {}
const char* methodName() const {
switch (fMethod) {
case Method::kImageAndOffset: return "imageAndOffset";
case Method::kDrawToCanvas: return "drawToCanvas";
case Method::kShader: return "asShader";
case Method::kClippedShader: return "asShaderClipped";
case Method::kStrictShader: return "strictShader";
}
SkUNREACHABLE;
}
std::pair<sk_sp<SkSpecialImage>, SkIPoint> resolve(const Context& ctx,
const FilterResult& image) const {
if (fMethod == Method::kImageAndOffset) {
SkIPoint origin;
sk_sp<SkSpecialImage> resolved = image.imageAndOffset(ctx, &origin);
return {resolved, origin};
} else {
if (ctx.desiredOutput().isEmpty()) {
return {nullptr, {}};
}
auto device = ctx.backend()->makeDevice(SkISize(ctx.desiredOutput().size()),
ctx.refColorSpace());
SkASSERT(device);
SkCanvas canvas{device};
canvas.clear(SK_ColorTRANSPARENT);
canvas.translate(-ctx.desiredOutput().left(), -ctx.desiredOutput().top());
if (fMethod > Method::kDrawToCanvas) {
sk_sp<SkShader> shader;
if (fMethod == Method::kShader) {
// asShader() applies layer bounds by resolving automatically
// (e.g. kDrawToCanvas), if sampleBounds is larger than the layer bounds. Since
// we want to test the unclipped shader version, pass in layerBounds() for
// sampleBounds and add a clip to the canvas instead.
canvas.clipIRect(SkIRect(image.layerBounds()));
shader = FilterResultTestAccess::AsShader(ctx, image, image.layerBounds());
} else if (fMethod == Method::kClippedShader) {
shader = FilterResultTestAccess::AsShader(ctx, image, ctx.desiredOutput());
} else {
shader = FilterResultTestAccess::StrictShader(ctx, image);
if (!shader) {
auto [pixels, origin] = this->resolve(
ctx.withNewDesiredOutput(image.layerBounds()), image);
shader = FilterResultTestAccess::StrictShader(
ctx, FilterResult(std::move(pixels), LayerSpace<SkIPoint>(origin)));
}
}
SkPaint paint;
paint.setShader(std::move(shader));
canvas.drawPaint(paint);
} else {
SkASSERT(fMethod == Method::kDrawToCanvas);
FilterResultTestAccess::Draw(ctx, device.get(), image,
/*preserveDeviceState=*/false);
}
return {device->snapSpecial(SkIRect::MakeWH(ctx.desiredOutput().width(),
ctx.desiredOutput().height())),
SkIPoint(ctx.desiredOutput().topLeft())};
}
}
private:
Method fMethod;
};
class TestRunner {
static constexpr SkColorType kColorType = kRGBA_8888_SkColorType;
using ResolveMethod = FilterResultImageResolver::Method;
public:
// Raster-backed TestRunner
TestRunner(skiatest::Reporter* reporter)
: fReporter(reporter)
, fBackend(skif::MakeRasterBackend(/*surfaceProps=*/{}, kColorType)) {}
// Ganesh-backed TestRunner
#if defined(SK_GANESH)
TestRunner(skiatest::Reporter* reporter, GrDirectContext* context)
: fReporter(reporter)
, fDirectContext(context)
, fBackend(skif::MakeGaneshBackend(sk_ref_sp(context),
kTopLeft_GrSurfaceOrigin,
/*surfaceProps=*/{},
kColorType)) {}
#endif
// Graphite-backed TestRunner
#if defined(SK_GRAPHITE)
TestRunner(skiatest::Reporter* reporter, skgpu::graphite::Recorder* recorder)
: fReporter(reporter)
, fRecorder(recorder)
, fBackend(skif::MakeGraphiteBackend(recorder, /*surfaceProps=*/{}, kColorType)) {}
#endif
// Let TestRunner be passed in to places that take a Reporter* or to REPORTER_ASSERT etc.
operator skiatest::Reporter*() const { return fReporter; }
skiatest::Reporter* operator->() const { return fReporter; }
skif::Backend* backend() const { return fBackend.get(); }
sk_sp<skif::Backend> refBackend() const { return fBackend; }
bool compareImages(const skif::Context& ctx,
SkSpecialImage* expectedImage,
SkIPoint expectedOrigin,
const FilterResult& actual,
float allowedPercentImageDiff,
int transparentCheckBorderTolerance) {
SkASSERT(expectedImage);
SkBitmap expectedBM = this->readPixels(expectedImage);
// Resolve actual using all 4 methods to ensure they are approximately equal to the expected
// (which is used as a proxy for being approximately equal to each other).
return this->compareImages(ctx, expectedBM, expectedOrigin, actual,
ResolveMethod::kImageAndOffset,
allowedPercentImageDiff, transparentCheckBorderTolerance) &&
this->compareImages(ctx, expectedBM, expectedOrigin, actual,
ResolveMethod::kDrawToCanvas,
allowedPercentImageDiff, transparentCheckBorderTolerance) &&
this->compareImages(ctx, expectedBM, expectedOrigin, actual,
ResolveMethod::kShader,
allowedPercentImageDiff, transparentCheckBorderTolerance) &&
this->compareImages(ctx, expectedBM, expectedOrigin, actual,
ResolveMethod::kClippedShader,
allowedPercentImageDiff, transparentCheckBorderTolerance);
}
bool validateOptimizedImage(const skif::Context& ctx, const FilterResult& actual) {
FilterResultImageResolver expectedResolver{ResolveMethod::kStrictShader};
auto [expectedImage, expectedOrigin] = expectedResolver.resolve(ctx, actual);
SkBitmap expectedBM = this->readPixels(expectedImage.get());
return this->compareImages(ctx, expectedBM, expectedOrigin, actual,
ResolveMethod::kImageAndOffset,
/*allowedPercentImageDiff=*/0.0f,
/*transparentCheckBorderTolerance=*/0);
}
sk_sp<SkSpecialImage> createSourceImage(SkISize size, sk_sp<SkColorSpace> colorSpace) {
sk_sp<SkDevice> sourceSurface = fBackend->makeDevice(size, std::move(colorSpace));
const SkColor colors[] = { SK_ColorMAGENTA,
SK_ColorRED,
SK_ColorYELLOW,
SK_ColorGREEN,
SK_ColorCYAN,
SK_ColorBLUE };
SkMatrix rotation = SkMatrix::RotateDeg(15.f, {size.width() / 2.f,
size.height() / 2.f});
SkCanvas canvas{sourceSurface};
canvas.clear(SK_ColorBLACK);
canvas.concat(rotation);
int color = 0;
SkRect coverBounds;
SkRect dstBounds = SkRect::Make(canvas.imageInfo().bounds());
SkAssertResult(SkMatrixPriv::InverseMapRect(rotation, &coverBounds, dstBounds));
float sz = size.width() <= 16.f || size.height() <= 16.f ? 2.f : 8.f;
for (float y = coverBounds.fTop; y < coverBounds.fBottom; y += sz) {
for (float x = coverBounds.fLeft; x < coverBounds.fRight; x += sz) {
SkPaint p;
p.setColor(colors[(color++) % std::size(colors)]);
canvas.drawRect(SkRect::MakeXYWH(x, y, sz, sz), p);
}
}
return sourceSurface->snapSpecial(SkIRect::MakeSize(size));
}
private:
bool compareImages(const skif::Context& ctx, const SkBitmap& expected, SkIPoint expectedOrigin,
const FilterResult& actual, ResolveMethod method,
float allowedPercentImageDiff, int transparentCheckBorderTolerance) {
FilterResultImageResolver resolver{method};
auto [actualImage, actualOrigin] = resolver.resolve(ctx, actual);
SkBitmap actualBM = this->readPixels(actualImage.get()); // empty if actualImage is null
TArray<SkIPoint> badPixels;
if (!this->compareBitmaps(expected, expectedOrigin, actualBM, actualOrigin,
allowedPercentImageDiff, transparentCheckBorderTolerance,
&badPixels)) {
if (!fLoggedErrorImage) {
SkDebugf("FilterResult comparison failed for method %s\n", resolver.methodName());
this->logBitmaps(expected, actualBM, badPixels);
fLoggedErrorImage = true;
}
return false;
} else if (kLogAllBitmaps) {
this->logBitmaps(expected, actualBM, badPixels);
}
return true;
}
bool compareBitmaps(const SkBitmap& expected,
SkIPoint expectedOrigin,
const SkBitmap& actual,
SkIPoint actualOrigin,
float allowedPercentImageDiff,
int transparentCheckBorderTolerance,
TArray<SkIPoint>* badPixels) {
SkIRect excludeTransparentCheck; // region in expectedBM that can be non-transparent
if (actual.empty()) {
// A null image in a FilterResult is equivalent to transparent black, so we should
// expect the contents of 'expectedImage' to be transparent black.
excludeTransparentCheck = SkIRect::MakeEmpty();
} else {
// The actual image bounds should be contained in the expected image's bounds.
SkIRect actualBounds = SkIRect::MakeXYWH(actualOrigin.x(), actualOrigin.y(),
actual.width(), actual.height());
SkIRect expectedBounds = SkIRect::MakeXYWH(expectedOrigin.x(), expectedOrigin.y(),
expected.width(), expected.height());
const bool contained = expectedBounds.contains(actualBounds);
REPORTER_ASSERT(fReporter, contained,
"actual image [%d %d %d %d] not contained within expected [%d %d %d %d]",
actualBounds.fLeft, actualBounds.fTop,
actualBounds.fRight, actualBounds.fBottom,
expectedBounds.fLeft, expectedBounds.fTop,
expectedBounds.fRight, expectedBounds.fBottom);
if (!contained) {
return false;
}
// The actual pixels should match fairly closely with the expected, allowing for minor
// differences from consolidating actions into a single render, etc.
int errorCount = 0;
SkIPoint offset = actualOrigin - expectedOrigin;
for (int y = 0; y < actual.height(); ++y) {
for (int x = 0; x < actual.width(); ++x) {
SkIPoint ep = {x + offset.x(), y + offset.y()};
SkColor4f expectedColor = expected.getColor4f(ep.fX, ep.fY);
SkColor4f actualColor = actual.getColor4f(x, y);
if (actualColor != expectedColor &&
!this->approxColor(this->boxFilter(actual, x, y),
this->boxFilter(expected, ep.fX, ep.fY))) {
badPixels->push_back(ep);
errorCount++;
}
}
}
const int totalCount = expected.width() * expected.height();
const float percentError = 100.f * errorCount / (float) totalCount;
const bool approxMatch = percentError <= allowedPercentImageDiff;
REPORTER_ASSERT(fReporter, approxMatch,
"%d pixels were too different from %d total (%f %% vs. %f %%)",
errorCount, totalCount, percentError, allowedPercentImageDiff);
if (!approxMatch) {
return false;
}
// The expected pixels outside of the actual bounds should be transparent, otherwise
// the actual image is not returning enough data.
excludeTransparentCheck = actualBounds.makeOffset(-expectedOrigin);
// Add per-test padding to the exclusion, which is used when there is upscaling in the
// expected image that bleeds beyond the layer bounds, but is hard to enforce in the
// simplified expectation rendering.
excludeTransparentCheck.outset(transparentCheckBorderTolerance,
transparentCheckBorderTolerance);
}
int badTransparencyCount = 0;
for (int y = 0; y < expected.height(); ++y) {
for (int x = 0; x < expected.width(); ++x) {
if (!excludeTransparentCheck.isEmpty() && excludeTransparentCheck.contains(x, y)) {
continue;
}
// If we are on the edge of the transparency exclusion bounds, allow pixels to be
// up to 2 off to account for sloppy GPU rendering (seen on some Android devices).
// This is still visually "transparent" and definitely make sure that
// off-transparency does not extend across the entire surface (tolerance = 0).
const bool onEdge = !excludeTransparentCheck.isEmpty() &&
excludeTransparentCheck.makeOutset(1, 1).contains(x, y);
if (!this->approxColor(expected.getColor4f(x, y), SkColors::kTransparent,
onEdge ? kAATolerance : 0.f)) {
badPixels->push_back({x, y});
badTransparencyCount++;
}
}
}
REPORTER_ASSERT(fReporter, badTransparencyCount == 0, "Unexpected non-transparent pixels");
return badTransparencyCount == 0;
}
bool approxColor(const SkColor4f& a,
const SkColor4f& b,
float tolerance = kRGBTolerance) const {
SkPMColor4f apm = a.premul();
SkPMColor4f bpm = b.premul();
// Calculate red-mean, a lowcost approximation of color difference that gives reasonable
// results for the types of acceptable differences resulting from collapsing compatible
// SkSamplingOptions or slightly different AA on shape boundaries.
// See https://www.compuphase.com/cmetric.htm
float r = (apm.fR + bpm.fR) / 2.f;
float dr = (apm.fR - bpm.fR);
float dg = (apm.fG - bpm.fG);
float db = (apm.fB - bpm.fB);
float delta = sqrt((2.f + r)*dr*dr + 4.f*dg*dg + (2.f + (1.f - r))*db*db);
return delta <= tolerance;
}
SkColor4f boxFilter(const SkBitmap& bm, int x, int y) const {
static constexpr int kKernelOffset = kKernelSize / 2;
SkPMColor4f sum = {0.f, 0.f, 0.f, 0.f};
float netWeight = 0.f;
for (int sy = y - kKernelOffset; sy <= y + kKernelOffset; ++sy) {
for (int sx = x - kKernelOffset; sx <= x + kKernelOffset; ++sx) {
float weight = kFuzzyKernel[sy - y + kKernelOffset][sx - x + kKernelOffset];
if (sx < 0 || sx >= bm.width() || sy < 0 || sy >= bm.height()) {
// Treat outside image as transparent black, this is necessary to get
// consistent comparisons between expected and actual images where the actual
// is cropped as tightly as possible.
netWeight += weight;
continue;
}
SkPMColor4f c = bm.getColor4f(sx, sy).premul() * weight;
sum.fR += c.fR;
sum.fG += c.fG;
sum.fB += c.fB;
sum.fA += c.fA;
netWeight += weight;
}
}
SkASSERT(netWeight > 0.f);
return sum.unpremul() * (1.f / netWeight);
}
SkBitmap readPixels(const SkSpecialImage* specialImage) const {
if (!specialImage) {
return SkBitmap(); // an empty bitmap
}
[[maybe_unused]] int srcX = specialImage->subset().fLeft;
[[maybe_unused]] int srcY = specialImage->subset().fTop;
SkImageInfo ii = SkImageInfo::Make(specialImage->dimensions(),
specialImage->colorInfo());
SkBitmap bm;
bm.allocPixels(ii);
#if defined(SK_GANESH)
if (fDirectContext) {
// Ganesh backed, just use the SkImage::readPixels API
SkASSERT(specialImage->isGaneshBacked());
sk_sp<SkImage> image = specialImage->asImage();
SkAssertResult(image->readPixels(fDirectContext, bm.pixmap(), srcX, srcY));
} else
#endif
#if defined(SK_GRAPHITE)
if (fRecorder) {
// Graphite backed, so use the private testing-only synchronous API
SkASSERT(specialImage->isGraphiteBacked());
auto view = skgpu::graphite::AsView(specialImage->asImage());
auto proxyII = ii.makeWH(view.width(), view.height());
SkAssertResult(fRecorder->priv().context()->priv().readPixels(
bm.pixmap(), view.proxy(), proxyII, srcX, srcY));
} else
#endif
{
// Assume it's raster backed, so use AsBitmap directly
SkAssertResult(SkSpecialImages::AsBitmap(specialImage, &bm));
}
return bm;
}
void logBitmaps(const SkBitmap& expected,
const SkBitmap& actual,
const TArray<SkIPoint>& badPixels) {
SkString expectedURL;
ToolUtils::BitmapToBase64DataURI(expected, &expectedURL);
SkDebugf("Expected:\n%s\n\n", expectedURL.c_str());
if (!actual.empty()) {
SkString actualURL;
ToolUtils::BitmapToBase64DataURI(actual, &actualURL);
SkDebugf("Actual:\n%s\n\n", actualURL.c_str());
} else {
SkDebugf("Actual: null (fully transparent)\n\n");
}
if (!badPixels.empty()) {
SkBitmap error = expected;
error.allocPixels();
SkAssertResult(expected.readPixels(error.pixmap()));
for (auto p : badPixels) {
error.erase(SkColors::kRed, SkIRect::MakeXYWH(p.fX, p.fY, 1, 1));
}
SkString markedURL;
ToolUtils::BitmapToBase64DataURI(error, &markedURL);
SkDebugf("Errors:\n%s\n\n", markedURL.c_str());
}
}
skiatest::Reporter* fReporter;
#if defined(SK_GANESH)
GrDirectContext* fDirectContext = nullptr;
#endif
#if defined(SK_GRAPHITE)
skgpu::graphite::Recorder* fRecorder = nullptr;
#endif
sk_sp<skif::Backend> fBackend;
bool fLoggedErrorImage = false; // only do this once per test runner
};
class TestCase {
public:
TestCase(TestRunner& runner,
std::string name,
float allowedPercentImageDiff=kDefaultMaxAllowedPercentImageDiff,
int transparentCheckBorderTolerance=0)
: fRunner(runner)
, fName(name)
, fAllowedPercentImageDiff(allowedPercentImageDiff)
, fTransparentCheckBorderTolerance(transparentCheckBorderTolerance)
, fSourceBounds(LayerSpace<SkIRect>::Empty())
, fDesiredOutput(LayerSpace<SkIRect>::Empty()) {}
TestCase& source(const SkIRect& bounds) {
fSourceBounds = LayerSpace<SkIRect>(bounds);
return *this;
}
TestCase& applyCrop(const SkIRect& crop, Expect expectation) {
return this->applyCrop(crop, SkTileMode::kDecal, expectation);
}
TestCase& applyCrop(const SkIRect& crop,
SkTileMode tileMode,
Expect expectation,
std::optional<SkTileMode> expectedTileMode = {},
std::optional<SkIRect> expectedBounds = {}) {
// Fill-in automated expectations, which is to equal 'tileMode' when not overridden.
if (!expectedTileMode) {
expectedTileMode = tileMode;
}
std::optional<LayerSpace<SkIRect>> expectedLayerBounds;
if (expectedBounds) {
expectedLayerBounds = LayerSpace<SkIRect>(*expectedBounds);
}
fActions.emplace_back(crop, tileMode, expectedLayerBounds, expectation,
this->getDefaultExpectedSampling(expectation),
*expectedTileMode,
this->getDefaultExpectedColorFilter(expectation));
return *this;
}
TestCase& applyTransform(const SkMatrix& matrix, Expect expectation) {
return this->applyTransform(matrix, FilterResult::kDefaultSampling, expectation);
}
TestCase& applyTransform(const SkMatrix& matrix,
const SkSamplingOptions& sampling,
Expect expectation,
std::optional<SkSamplingOptions> expectedSampling = {}) {
// Fill-in automated expectations, which is simply that if it's not explicitly provided we
// assume the result's sampling equals what was passed to applyTransform().
if (!expectedSampling.has_value()) {
expectedSampling = sampling;
}
fActions.emplace_back(matrix, sampling, expectation, *expectedSampling,
this->getDefaultExpectedTileMode(expectation,
/*cfAffectsTransparency=*/false),
this->getDefaultExpectedColorFilter(expectation));
return *this;
}
TestCase& applyColorFilter(sk_sp<SkColorFilter> colorFilter,
Expect expectation,
std::optional<sk_sp<SkColorFilter>> expectedColorFilter = {}) {
// The expected color filter is the composition of the default expectation (e.g. last
// color filter or null for a new image) and the new 'colorFilter'. Compose() automatically
// returns 'colorFilter' if the inner filter is null.
if (!expectedColorFilter.has_value()) {
expectedColorFilter = SkColorFilters::Compose(
colorFilter, this->getDefaultExpectedColorFilter(expectation));
}
const bool affectsTransparent = as_CFB(colorFilter)->affectsTransparentBlack();
fActions.emplace_back(std::move(colorFilter), expectation,
this->getDefaultExpectedSampling(expectation),
this->getDefaultExpectedTileMode(expectation, affectsTransparent),
std::move(*expectedColorFilter));
return *this;
}
TestCase& rescale(SkSize scale,
Expect expectation,
std::optional<SkTileMode> expectedTileMode = {}) {
SkASSERT(!fActions.empty());
if (!expectedTileMode) {
expectedTileMode = this->getDefaultExpectedTileMode(expectation,
/*cfAffectsTransparency=*/false);
}
fActions.emplace_back(skif::LayerSpace<SkSize>(scale), expectation,
this->getDefaultExpectedSampling(expectation),
*expectedTileMode,
this->getDefaultExpectedColorFilter(expectation));
return *this;
}
void run(const SkIRect& requestedOutput) const {
skiatest::ReporterContext caseLabel(fRunner, fName);
this->run(requestedOutput, /*backPropagateDesiredOutput=*/true);
this->run(requestedOutput, /*backPropagateDesiredOutput=*/false);
}
void run(const SkIRect& requestedOutput, bool backPropagateDesiredOutput) const {
SkASSERT(!fActions.empty()); // It's a bad test case if there aren't any actions
skiatest::ReporterContext backPropagate(
fRunner, SkStringPrintf("backpropagate output: %d", backPropagateDesiredOutput));
auto desiredOutput = LayerSpace<SkIRect>(requestedOutput);
std::vector<LayerSpace<SkIRect>> desiredOutputs;
desiredOutputs.resize(fActions.size(), desiredOutput);
if (!backPropagateDesiredOutput) {
// Set the desired output to be equal to the expected output so that there is no
// further restriction of what's computed for early actions to then be ruled out by
// subsequent actions.
auto inputBounds = fSourceBounds;
for (int i = 0; i < (int) fActions.size() - 1; ++i) {
desiredOutputs[i] = fActions[i].expectedBounds(inputBounds);
// If the output for the ith action is infinite, leave it for now and expand the
// input bounds for action i+1. The infinite bounds will be replaced by the
// back-propagated desired output of the next action.
if (SkIRect(desiredOutputs[i]) == SkRectPriv::MakeILarge()) {
inputBounds.outset(LayerSpace<SkISize>({25, 25}));
} else {
inputBounds = desiredOutputs[i];
}
}
}
// Fill out regular back-propagated desired outputs and cleanup infinite outputs
for (int i = (int) fActions.size() - 2; i >= 0; --i) {
if (backPropagateDesiredOutput ||
SkIRect(desiredOutputs[i]) == SkRectPriv::MakeILarge()) {
desiredOutputs[i] = fActions[i+1].requiredInput(desiredOutputs[i+1]);
}
}
// Create the source image
sk_sp<SkColorSpace> colorSpace = SkColorSpace::MakeSRGB();
FilterResult source;
if (!fSourceBounds.isEmpty()) {
source = FilterResult(fRunner.createSourceImage(SkISize(fSourceBounds.size()),
colorSpace),
fSourceBounds.topLeft());
}
Context baseContext{fRunner.refBackend(),
skif::Mapping{SkMatrix::I()},
skif::LayerSpace<SkIRect>::Empty(),
source,
colorSpace.get(),
/*stats=*/nullptr};
// Applying modifiers to FilterResult might produce a new image, but hopefully it's
// able to merge properties and even re-order operations to minimize the number of offscreen
// surfaces that it creates. To validate that this is producing an equivalent image, we
// track what to expect by rendering each action every time without any optimization.
sk_sp<SkSpecialImage> expectedImage = source.refImage();
LayerSpace<SkIPoint> expectedOrigin = source.layerBounds().topLeft();
// The expected image can't ever be null, so we produce a transparent black image instead.
if (!expectedImage) {
sk_sp<SkDevice> expectedSurface = fRunner.backend()->makeDevice({1, 1}, colorSpace);
clear_device(expectedSurface.get());
expectedImage = expectedSurface->snapSpecial(SkIRect::MakeWH(1, 1));
expectedOrigin = LayerSpace<SkIPoint>({0, 0});
}
SkASSERT(expectedImage);
// Apply each action and validate, from first to last action
for (int i = 0; i < (int) fActions.size(); ++i) {
skiatest::ReporterContext actionLabel(fRunner, SkStringPrintf("action %d", i));
Stats stats;
auto ctx = baseContext.withNewDesiredOutput(desiredOutputs[i]);
FilterResultTestAccess::TrackStats(&ctx, &stats);
FilterResult output = fActions[i].apply(ctx, source);
// Validate consistency of the output
REPORTER_ASSERT(fRunner, SkToBool(output.image()) == !output.layerBounds().isEmpty());
LayerSpace<SkIRect> expectedBounds = fActions[i].expectedBounds(source.layerBounds());
Expect correctedExpectation = fActions[i].expectation();
if (SkIRect(expectedBounds) == SkRectPriv::MakeILarge()) {
// An expected image filling out to infinity should have an actual image that
// fills the desired output.
expectedBounds = desiredOutputs[i];
if (desiredOutputs[i].isEmpty()) {
correctedExpectation = Expect::kEmptyImage;
}
} else if (!expectedBounds.intersect(desiredOutputs[i])) {
// Test cases should provide image expectations for the case where desired output
// is not back-propagated. When desired output is back-propagated, it can lead to
// earlier actions becoming empty actions.
REPORTER_ASSERT(fRunner, fActions[i].expectation() == Expect::kEmptyImage ||
backPropagateDesiredOutput);
expectedBounds = LayerSpace<SkIRect>::Empty();
correctedExpectation = Expect::kEmptyImage;
}
int numOffscreenSurfaces = fActions[i].expectedOffscreenSurfaces();
int actualShaderDraws = stats.fNumShaderBasedTilingDraws + stats.fNumShaderClampedDraws;
int expectedShaderTiledDraws = 0;
bool actualNewImage = output.image() &&
(!source.image() || output.image()->uniqueID() != source.image()->uniqueID());
switch(correctedExpectation) {
case Expect::kNewImage:
REPORTER_ASSERT(fRunner, actualNewImage);
if (source && !source.image()->isExactFit()) {
expectedShaderTiledDraws = numOffscreenSurfaces;
}
break;
case Expect::kDeferredImage:
REPORTER_ASSERT(fRunner, !actualNewImage && output.image());
break;
case Expect::kEmptyImage:
REPORTER_ASSERT(fRunner, !actualNewImage && !output.image());
break;
}
// Verify stats behavior for the current action
REPORTER_ASSERT(fRunner, numOffscreenSurfaces == stats.fNumOffscreenSurfaces,
"expected %d, got %d",
numOffscreenSurfaces, stats.fNumOffscreenSurfaces);
REPORTER_ASSERT(fRunner, actualShaderDraws <= expectedShaderTiledDraws,
"expected %d+%d <= %d",
stats.fNumShaderBasedTilingDraws, stats.fNumShaderClampedDraws,
expectedShaderTiledDraws);
const bool rescaling = fActions[i].isRescaling();
REPORTER_ASSERT(fRunner, stats.fNumShaderBasedTilingDraws == 0 ||
FilterResultTestAccess::IsShaderTilingExpected(
ctx, source, rescaling));
REPORTER_ASSERT(fRunner, stats.fNumShaderClampedDraws == 0 ||
FilterResultTestAccess::IsShaderClampingExpected(
ctx, source, rescaling));
// Validate layer bounds and sampling when we expect a new or deferred image
if (output.image()) {
auto actualBounds = output.layerBounds();
// A deferred action doesn't have to crop its layer bounds to the desired output to
// preserve accuracy of later bounds analysis. New images however should restrict
// themselves to the desired output to minimize memory of the surface. The exception
// is a new image for applyTransform() because the new transform is deferred to the
// resolved image, which can make its layer bounds larger than the desired output.
if (correctedExpectation == Expect::kDeferredImage ||
!FilterResultTestAccess::IsIntegerTransform(output)) {
REPORTER_ASSERT(fRunner, actualBounds.intersect(desiredOutputs[i]));
}
REPORTER_ASSERT(fRunner, !expectedBounds.isEmpty());
REPORTER_ASSERT(fRunner, SkIRect(actualBounds) == SkIRect(expectedBounds));
REPORTER_ASSERT(fRunner, output.sampling() == fActions[i].expectedSampling());
REPORTER_ASSERT(fRunner, output.tileMode() == fActions[i].expectedTileMode());
REPORTER_ASSERT(fRunner, colorfilter_equals(output.colorFilter(),
fActions[i].expectedColorFilter()));
if (actualShaderDraws < expectedShaderTiledDraws ||
(source.tileMode() != SkTileMode::kClamp && stats.fNumShaderClampedDraws > 0)) {
// Some tile draws were optimized to HW draws, or some tile draws were reduced
// to shader-clamped draws, so compare the output to a non-optimized image.
REPORTER_ASSERT(fRunner, fRunner.validateOptimizedImage(ctx, output));
}
}
expectedImage = fActions[i].renderExpectedImage(ctx,
std::move(expectedImage),
expectedOrigin,
desiredOutputs[i]);
expectedOrigin = desiredOutputs[i].topLeft();
if (!fRunner.compareImages(ctx,
expectedImage.get(),
SkIPoint(expectedOrigin),
output,
fAllowedPercentImageDiff,
fTransparentCheckBorderTolerance)) {
// If one iteration is incorrect, its failures will likely cascade to further
// actions so end now as the test has failed.
break;
}
source = output;
}
}
private:
// By default an action that doesn't define its own sampling options will not change sampling
// unless it produces a new image. Otherwise it inherits the prior action's expectation.
SkSamplingOptions getDefaultExpectedSampling(Expect expectation) const {
if (expectation != Expect::kDeferredImage || fActions.empty()) {
return FilterResult::kDefaultSampling;
} else {
return fActions[fActions.size() - 1].expectedSampling();
}
}
// By default an action that doesn't define its own tiling will not change the tiling, unless it
// produces a new image, at which point it becomes kDecal again.
SkTileMode getDefaultExpectedTileMode(Expect expectation, bool cfAffectsTransparency) const {
if (expectation == Expect::kNewImage && cfAffectsTransparency) {
return SkTileMode::kClamp;
} else if (expectation != Expect::kDeferredImage || fActions.empty()) {
return SkTileMode::kDecal;
} else {
return fActions[fActions.size() - 1].expectedTileMode();
}
}
// By default an action that doesn't define its own color filter will not change filtering,
// unless it produces a new image. Otherwise it inherits the prior action's expectations.
sk_sp<SkColorFilter> getDefaultExpectedColorFilter(Expect expectation) const {
if (expectation != Expect::kDeferredImage || fActions.empty()) {
return nullptr;
} else {
return sk_ref_sp(fActions[fActions.size() - 1].expectedColorFilter());
}
}
TestRunner& fRunner;
std::string fName;
float fAllowedPercentImageDiff;
int fTransparentCheckBorderTolerance;
// Used to construct an SkSpecialImage of the given size/location filled with the known pattern.
LayerSpace<SkIRect> fSourceBounds;
// The intended area to fill with the result, controlled by outside factors (e.g. clip bounds)
LayerSpace<SkIRect> fDesiredOutput;
std::vector<ApplyAction> fActions;
};
// ----------------------------------------------------------------------------
// Utilities to create color filters for the unit tests
sk_sp<SkColorFilter> alpha_modulate(float v) {
// dst-in blending with src = (1,1,1,v) = dst * v
auto cf = SkColorFilters::Blend({1.f,1.f,1.f,v}, /*colorSpace=*/nullptr, SkBlendMode::kDstIn);
SkASSERT(cf && !as_CFB(cf)->affectsTransparentBlack());
return cf;
}
sk_sp<SkColorFilter> affect_transparent(SkColor4f color) {
auto cf = SkColorFilters::Blend(color, /*colorSpace=*/nullptr, SkBlendMode::kPlus);
SkASSERT(cf && as_CFB(cf)->affectsTransparentBlack());
return cf;
}
// ----------------------------------------------------------------------------
// TODO(skbug.com/14607) - Run FilterResultTests on Dawn and ANGLE backends, too
#if defined(SK_GANESH)
#define DEF_GANESH_TEST_SUITE(name, ctsEnforcement) \
DEF_GANESH_TEST_FOR_CONTEXTS(FilterResult_ganesh_##name, \
skgpu::IsNativeBackend, \
r, \
ctxInfo, \
nullptr, \
ctsEnforcement) { \
TestRunner runner(r, ctxInfo.directContext()); \
test_suite_##name(runner); \
}
#else
#define DEF_GANESH_TEST_SUITE(name) // do nothing
#endif
#if defined(SK_GRAPHITE)
#define DEF_GRAPHITE_TEST_SUITE(name, ctsEnforcement) \
DEF_CONDITIONAL_GRAPHITE_TEST_FOR_ALL_CONTEXTS(FilterResult_graphite_##name, \
skgpu::IsNativeBackend, \
r, \
context, \
testContext, \
true, \
ctsEnforcement) { \
using namespace skgpu::graphite; \
auto recorder = context->makeRecorder(); \
TestRunner runner(r, recorder.get()); \
test_suite_##name(runner); \
std::unique_ptr<Recording> recording = recorder->snap(); \
if (!recording) { \
ERRORF(r, "Failed to make recording"); \
return; \
} \
InsertRecordingInfo insertInfo; \
insertInfo.fRecording = recording.get(); \
context->insertRecording(insertInfo); \
testContext->syncedSubmit(context); \
}
#else
#define DEF_GRAPHITE_TEST_SUITE(name) // do nothing
#endif
#define DEF_TEST_SUITE(name, runner, ganeshCtsEnforcement, graphiteCtsEnforcement) \
static void test_suite_##name(TestRunner&); \
/* TODO(b/274901800): Uncomment to enable Graphite test execution. */ \
/* DEF_GRAPHITE_TEST_SUITE(name, graphiteCtsEnforcement) */ \
DEF_GANESH_TEST_SUITE(name, ganeshCtsEnforcement) \
DEF_TEST(FilterResult_raster_##name, reporter) { \
TestRunner runner(reporter); \
test_suite_##name(runner); \
} \
void test_suite_##name(TestRunner& runner)
// ----------------------------------------------------------------------------
// Empty input/output tests
DEF_TEST_SUITE(EmptySource, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
// This is testing that an empty input image is handled by the applied actions without having
// to generate new images, or that it can produce a new image from nothing when it affects
// transparent black.
for (SkTileMode tm : kTileModes) {
TestCase(r, "applyCrop() to empty source")
.source(SkIRect::MakeEmpty())
.applyCrop({0, 0, 10, 10}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 20, 20});
}
TestCase(r, "applyTransform() to empty source")
.source(SkIRect::MakeEmpty())
.applyTransform(SkMatrix::Translate(10.f, 10.f), Expect::kEmptyImage)
.run(/*requestedOutput=*/{10, 10, 20, 20});
TestCase(r, "applyColorFilter() to empty source")
.source(SkIRect::MakeEmpty())
.applyColorFilter(alpha_modulate(0.5f), Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
TestCase(r, "Transparency-affecting color filter overrules empty source")
.source(SkIRect::MakeEmpty())
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kNewImage,
/*expectedColorFilter=*/nullptr) // CF applied ASAP to make a new img
.run(/*requestedOutput=*/{0, 0, 10, 10});
}
DEF_TEST_SUITE(EmptyDesiredOutput, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
// This is testing that an empty requested output is propagated through the applied actions so
// that no actual images are generated.
for (SkTileMode tm : kTileModes) {
TestCase(r, "applyCrop() + empty output becomes empty")
.source({0, 0, 10, 10})
.applyCrop({2, 2, 8, 8}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/SkIRect::MakeEmpty());
}
TestCase(r, "applyTransform() + empty output becomes empty")
.source({0, 0, 10, 10})
.applyTransform(SkMatrix::RotateDeg(10.f), Expect::kEmptyImage)
.run(/*requestedOutput=*/SkIRect::MakeEmpty());
TestCase(r, "applyColorFilter() + empty output becomes empty")
.source({0, 0, 10, 10})
.applyColorFilter(alpha_modulate(0.5f), Expect::kEmptyImage)
.run(/*requestedOutput=*/SkIRect::MakeEmpty());
TestCase(r, "Transpency-affecting color filter + empty output is empty")
.source({0, 0, 10, 10})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kEmptyImage)
.run(/*requestedOutput=*/SkIRect::MakeEmpty());
}
// ----------------------------------------------------------------------------
// applyCrop() tests
DEF_TEST_SUITE(Crop, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
// This is testing all the combinations of how the src, crop, and requested output rectangles
// can interact while still resulting in a deferred image. The exception is non-decal tile
// modes where the crop rect includes transparent pixels not filled by the source, which
// requires a new image to ensure tiling matches the crop geometry.
for (SkTileMode tm : kTileModes) {
const Expect nonDecalExpectsNewImage = tm == SkTileMode::kDecal ? Expect::kDeferredImage
: Expect::kNewImage;
TestCase(r, "applyCrop() contained in source and output")
.source({0, 0, 20, 20})
.applyCrop({8, 8, 12, 12}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{4, 4, 16, 16});
TestCase(r, "applyCrop() contained in source, intersects output")
.source({0, 0, 20, 20})
.applyCrop({4, 4, 12, 12}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{8, 8, 16, 16});
TestCase(r, "applyCrop() intersects source, contained in output")
.source({10, 10, 20, 20})
.applyCrop({4, 4, 16, 16}, tm, nonDecalExpectsNewImage)
.run(/*requestedOutput=*/{0, 0, 20, 20});
TestCase(r, "applyCrop() intersects source and output")
.source({0, 0, 10, 10})
.applyCrop({5, -5, 15, 5}, tm, nonDecalExpectsNewImage)
.run(/*requestedOutput=*/{7, -2, 12, 8});
TestCase(r, "applyCrop() contains source, intersects output")
.source({4, 4, 16, 16})
.applyCrop({0, 0, 20, 20}, tm, nonDecalExpectsNewImage)
.run(/*requestedOutput=*/{-5, -5, 18, 18});
// In these cases, cropping with a non-decal tile mode can be discarded because the output
// bounds are entirely within the crop so no tiled edges would be visible.
TestCase(r, "applyCrop() intersects source, contains output")
.source({0, 0, 20, 20})
.applyCrop({-5, 5, 25, 15}, tm, Expect::kDeferredImage, SkTileMode::kDecal)
.run(/*requestedOutput=*/{0, 5, 20, 15});
TestCase(r, "applyCrop() contains source and output")
.source({0, 0, 10, 10})
.applyCrop({-5, -5, 15, 15}, tm, Expect::kDeferredImage, SkTileMode::kDecal)
.run(/*requestedOutput=*/{1, 1, 9, 9});
}
}
DEF_TEST_SUITE(CropDisjointFromSourceAndOutput, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
// This tests all the combinations of src, crop, and requested output rectangles that result in
// an empty image without any of the rectangles being empty themselves. The exception is for
// non-decal tile modes when the source and crop still intersect. In that case the non-empty
// content is tiled into the disjoint output rect, producing a non-empty image.
for (SkTileMode tm : kTileModes) {
TestCase(r, "applyCrop() disjoint from source, intersects output")
.source({0, 0, 10, 10})
.applyCrop({11, 11, 20, 20}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 15, 15});
TestCase(r, "applyCrop() disjoint from source, intersects output disjoint from source")
.source({0, 0, 10, 10})
.applyCrop({11, 11, 20, 20}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{12, 12, 18, 18});
TestCase(r, "applyCrop() disjoint from source and output")
.source({0, 0, 10, 10})
.applyCrop({12, 12, 18, 18}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{-1, -1, 11, 11});
TestCase(r, "applyCrop() disjoint from source and output disjoint from source")
.source({0, 0, 10, 10})
.applyCrop({-10, 10, -1, -1}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{11, 11, 20, 20});
// When the source and crop intersect but are disjoint from the output, the behavior depends
// on the tile mode. For periodic tile modes, certain geometries can still be deferred by
// conversion to a transform, but to keep expectations simple we pick bounds such that the
// tiling can't be dropped. See PeriodicTileCrops for other scenarios.
Expect nonDecalExpectsImage = tm == SkTileMode::kDecal ? Expect::kEmptyImage :
tm == SkTileMode::kClamp ? Expect::kDeferredImage
: Expect::kNewImage;
TestCase(r, "applyCrop() intersects source, disjoint from output disjoint from source")
.source({0, 0, 10, 10})
.applyCrop({-5, -5, 5, 5}, tm, nonDecalExpectsImage)
.run(/*requestedOutput=*/{12, 12, 18, 18});
TestCase(r, "applyCrop() intersects source, disjoint from output")
.source({0, 0, 10, 10})
.applyCrop({-5, -5, 5, 5}, tm, nonDecalExpectsImage)
.run(/*requestedOutput=*/{6, 6, 18, 18});
}
}
DEF_TEST_SUITE(EmptyCrop, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "applyCrop() is empty")
.source({0, 0, 10, 10})
.applyCrop(SkIRect::MakeEmpty(), tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
TestCase(r, "applyCrop() emptiness propagates")
.source({0, 0, 10, 10})
.applyCrop({1, 1, 9, 9}, tm, Expect::kDeferredImage)
.applyCrop(SkIRect::MakeEmpty(), tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
}
}
DEF_TEST_SUITE(DisjointCrops, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "Disjoint applyCrop() after kDecal become empty")
.source({0, 0, 10, 10})
.applyCrop({0, 0, 4, 4}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyCrop({6, 6, 10, 10}, tm, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
if (tm != SkTileMode::kDecal) {
TestCase(r, "Disjoint tiling applyCrop() before kDecal is not empty and combines")
.source({0, 0, 10, 10})
.applyCrop({0, 0, 4, 4}, tm, Expect::kDeferredImage)
.applyCrop({6, 6, 10, 10}, SkTileMode::kDecal, Expect::kDeferredImage, tm)
.run(/*requestedOutput=*/{0, 0, 10, 10});
TestCase(r, "Disjoint non-decal applyCrops() are not empty")
.source({0, 0, 10, 10})
.applyCrop({0, 0, 4, 4}, tm, Expect::kDeferredImage)
.applyCrop({6, 6, 10, 10}, tm, tm == SkTileMode::kClamp ? Expect::kDeferredImage
: Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
}
}
}
DEF_TEST_SUITE(IntersectingCrops, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "Decal applyCrop() always combines with any other crop")
.source({0, 0, 20, 20})
.applyCrop({5, 5, 15, 15}, tm, Expect::kDeferredImage)
.applyCrop({10, 10, 20, 20}, SkTileMode::kDecal, Expect::kDeferredImage, tm)
.run(/*requestedOutput=*/{0, 0, 20, 20});
if (tm != SkTileMode::kDecal) {
TestCase(r, "Decal applyCrop() before non-decal crop requires new image")
.source({0, 0, 20, 20})
.applyCrop({5, 5, 15, 15}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyCrop({10, 10, 20, 20}, tm, Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 20, 20});
TestCase(r, "Consecutive non-decal crops combine if both are clamp")
.source({0, 0, 20, 20})
.applyCrop({5, 5, 15, 15}, tm, Expect::kDeferredImage)
.applyCrop({10, 10, 20, 20}, tm,
tm == SkTileMode::kClamp ? Expect::kDeferredImage
: Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 20, 20});
}
}
}
DEF_TEST_SUITE(PeriodicTileCrops, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : {SkTileMode::kRepeat, SkTileMode::kMirror}) {
// In these tests, the crop periodically tiles such that it covers the desired output so
// the prior image can be simply transformed.
TestCase(r, "Periodic applyCrop() becomes a transform")
.source({0, 0, 20, 20})
.applyCrop({5, 5, 15, 15}, tm, Expect::kDeferredImage,
/*expectedTileMode=*/SkTileMode::kDecal)
.run(/*requestedOutput=*/{25, 25, 35, 35});
TestCase(r, "Periodic applyCrop() with partial transparency still becomes a transform")
.source({0, 0, 20, 20})
.applyCrop({-5, -5, 15, 15}, tm, Expect::kDeferredImage,
/*expectedTileMode=*/SkTileMode::kDecal,
/*expectedBounds=*/tm == SkTileMode::kRepeat ? SkIRect{20,20,35,35}
: SkIRect{15,15,30,30})
.run(/*requestedOutput*/{15, 15, 35, 35});
TestCase(r, "Periodic applyCrop() after complex transform can still simplify")
.source({0, 0, 20, 20})
.applyTransform(SkMatrix::RotateDeg(15.f, {10.f, 10.f}), Expect::kDeferredImage)
.applyCrop({-5, -5, 25, 25}, tm, Expect::kDeferredImage,
/*expectedTileMode=*/SkTileMode::kDecal,
/*expectedBounds*/SkIRect{57,57,83,83}) // source+15 degree rotation
.run(/*requestedOutput=*/{55,55,85,85});
// In these tests, the crop's periodic boundary intersects with the output so it should not
// simplify to just a transform.
TestCase(r, "Periodic applyCrop() with visible edge does not become a transform")
.source({0, 0, 20, 20})
.applyCrop({5, 5, 15, 15}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{10, 10, 20, 20});
TestCase(r, "Periodic applyCrop() with visible edge and transparency creates new image")
.source({0, 0, 20, 20})
.applyCrop({-5, -5, 15, 15}, tm, Expect::kNewImage)
.run(/*requestedOutput=*/{10, 10, 20, 20});
TestCase(r, "Periodic applyCropp() with visible edge and complex transform creates image")
.source({0, 0, 20, 20})
.applyTransform(SkMatrix::RotateDeg(15.f, {10.f, 10.f}), Expect::kDeferredImage)
.applyCrop({-5, -5, 25, 25}, tm, Expect::kNewImage)
.run(/*requestedOutput=*/{20, 20, 50, 50});
}
}
DEF_TEST_SUITE(DecalThenClamp, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
TestCase(r, "Decal then clamp crop uses 1px buffer around intersection")
.source({0, 0, 20, 20})
.applyCrop({3, 3, 17, 17}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyCrop({3, 3, 20, 20}, SkTileMode::kClamp, Expect::kNewImage, SkTileMode::kClamp)
.run(/*requestedOutput=*/{0, 0, 20, 20});
TestCase(r, "Decal then clamp crop uses 1px buffer around intersection, w/ alpha color filter")
.source({0, 0, 20, 20})
.applyCrop({3, 3, 17, 17}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kCyan), Expect::kDeferredImage)
.applyCrop({0, 0, 17, 17}, SkTileMode::kClamp, Expect::kNewImage, SkTileMode::kClamp)
.run(/*requestedOutput=*/{0, 0, 20, 20});
}
// ----------------------------------------------------------------------------
// applyTransform() tests
DEF_TEST_SUITE(Transform, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
TestCase(r, "applyTransform() integer translate")
.source({0, 0, 10, 10})
.applyTransform(SkMatrix::Translate(5, 5), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
TestCase(r, "applyTransform() fractional translate")
.source({0, 0, 10, 10})
.applyTransform(SkMatrix::Translate(1.5f, 3.24f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 10, 10});
TestCase(r, "applyTransform() scale")
.source({0, 0, 24, 24})
.applyTransform(SkMatrix::Scale(2.2f, 3.1f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-16, -16, 96, 96});
// NOTE: complex is anything beyond a scale+translate. See SkImageFilter_Base::MatrixCapability.
TestCase(r, "applyTransform() with complex transform")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(10.f, {4.f, 4.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
}
DEF_TEST_SUITE(CompatibleSamplingConcatsTransforms, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "linear + linear combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "equiv. bicubics combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "linear + bicubic becomes bicubic")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "bicubic + linear becomes bicubic")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage,
/*expectedSampling=*/SkCubicResampler::Mitchell())
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "aniso picks max level to combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(4.f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(2.f), Expect::kDeferredImage,
/*expectedSampling=*/SkSamplingOptions::Aniso(4.f))
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "aniso picks max level to combine (other direction)")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(2.f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(4.f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "linear + aniso becomes aniso")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(2.f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "aniso + linear stays aniso")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(4.f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage,
/*expectedSampling=*/SkSamplingOptions::Aniso(4.f))
.run(/*requestedOutput=*/{0, 0, 16, 16});
// TODO: Add cases for mipmapping once that becomes relevant (SkSpecialImage does not have
// mipmaps right now).
}
DEF_TEST_SUITE(IncompatibleSamplingResolvesImages, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "different bicubics do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::CatmullRom(), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "nearest + linear do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kNearest, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "linear + nearest do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kLinear, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kNearest, Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "bicubic + aniso do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(4.f), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "aniso + bicubic do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkSamplingOptions::Aniso(4.f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "nearest + nearest do not combine")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kNearest, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkFilterMode::kNearest, Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
}
DEF_TEST_SUITE(IntegerOffsetIgnoresNearestSampling, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
// Bicubic is used here to reflect that it should use the non-NN sampling and just needs to be
// something other than the default to detect that it got carried through.
TestCase(r, "integer translate+NN then bicubic combines")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::Translate(2, 2),
SkFilterMode::kNearest, Expect::kDeferredImage,
FilterResult::kDefaultSampling)
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "bicubic then integer translate+NN combines")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::RotateDeg(2.f, {4.f, 4.f}),
SkCubicResampler::Mitchell(), Expect::kDeferredImage)
.applyTransform(SkMatrix::Translate(2, 2),
SkFilterMode::kNearest, Expect::kDeferredImage,
/*expectedSampling=*/SkCubicResampler::Mitchell())
.run(/*requestedOutput=*/{0, 0, 16, 16});
}
// ----------------------------------------------------------------------------
// applyTransform() interacting with applyCrop()
DEF_TEST_SUITE(TransformBecomesEmpty, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "Transform moves src image outside of requested output")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::Translate(10.f, 10.f), Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 8, 8});
TestCase(r, "Transform moves src image outside of crop")
.source({0, 0, 8, 8})
.applyTransform(SkMatrix::Translate(10.f, 10.f), Expect::kDeferredImage)
.applyCrop({2, 2, 6, 6}, Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 20, 20});
TestCase(r, "Transform moves cropped image outside of requested output")
.source({0, 0, 8, 8})
.applyCrop({1, 1, 4, 4}, Expect::kDeferredImage)
.applyTransform(SkMatrix::Translate(-5.f, -5.f), Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 8, 8});
}
DEF_TEST_SUITE(TransformAndCrop, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
TestCase(r, "Crop after transform can always apply")
.source({0, 0, 16, 16})
.applyTransform(SkMatrix::RotateDeg(45.f, {3.f, 4.f}), Expect::kDeferredImage)
.applyCrop({2, 2, 15, 15}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
// TODO: Expand this test case to be arbitrary float S+T transforms when FilterResult tracks
// both a srcRect and dstRect.
TestCase(r, "Crop after translate is lifted to image subset")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::Translate(12.f, 8.f), Expect::kDeferredImage)
.applyCrop({16, 16, 24, 24}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {16.f, 16.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transform after unlifted crop triggers new image")
.source({0, 0, 16, 16})
.applyTransform(SkMatrix::RotateDeg(45.f, {8.f, 8.f}), Expect::kDeferredImage)
.applyCrop({1, 1, 15, 15}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(-10.f, {8.f, 4.f}), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "Transform after unlifted crop with interior output does not trigger new image")
.source({0, 0, 16, 16})
.applyTransform(SkMatrix::RotateDeg(45.f, {8.f, 8.f}), Expect::kDeferredImage)
.applyCrop({1, 1, 15, 15}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(-10.f, {8.f, 4.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{4, 4, 12, 12});
TestCase(r, "Translate after unlifted crop does not trigger new image")
.source({0, 0, 16, 16})
.applyTransform(SkMatrix::RotateDeg(5.f, {8.f, 8.f}), Expect::kDeferredImage)
.applyCrop({2, 2, 14, 14}, Expect::kDeferredImage)
.applyTransform(SkMatrix::Translate(4.f, 6.f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "Transform after large no-op crop does not trigger new image")
.source({0, 0, 64, 64})
.applyTransform(SkMatrix::RotateDeg(45.f, {32.f, 32.f}), Expect::kDeferredImage)
.applyCrop({-64, -64, 128, 128}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(-30.f, {32.f, 32.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 64, 64});
}
DEF_TEST_SUITE(TransformAndTile, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
// Test interactions of non-decal tile modes and transforms
for (SkTileMode tm : kTileModes) {
if (tm == SkTileMode::kDecal) {
continue;
}
TestCase(r, "Transform after tile mode does not trigger new image")
.source({0, 0, 64, 64})
.applyCrop({2, 2, 32, 32}, tm, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(20.f, {16.f, 8.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 64, 64});
TestCase(r, "Integer transform before tile mode does not trigger new image")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::Translate(16.f, 16.f), Expect::kDeferredImage)
.applyCrop({20, 20, 40, 40}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 64, 64});
TestCase(r, "Non-integer transform before tile mode triggers new image")
.source({0, 0, 50, 40})
.applyTransform(SkMatrix::RotateDeg(-30.f, {20.f, 10.f}), Expect::kDeferredImage)
.applyCrop({10, 10, 30, 30}, tm, Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 50, 50});
TestCase(r, "Non-integer transform before tiling defers image if edges are hidden")
.source({0, 0, 64, 64})
.applyTransform(SkMatrix::RotateDeg(45.f, {32.f, 32.f}), Expect::kDeferredImage)
.applyCrop({10, 10, 50, 50}, tm, Expect::kDeferredImage,
/*expectedTileMode=*/SkTileMode::kDecal)
.run(/*requestedOutput=*/{11, 11, 49, 49});
}
}
// ----------------------------------------------------------------------------
// applyColorFilter() and interactions with transforms/crops
DEF_TEST_SUITE(ColorFilter, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
TestCase(r, "applyColorFilter() defers image")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "applyColorFilter() composes with other color filters")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transparency-affecting color filter fills output")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-8, -8, 32, 32});
// Since there is no cropping between the composed color filters, transparency-affecting CFs
// can still compose together.
TestCase(r, "Transparency-affecting composition fills output (ATBx2)")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-8, -8, 32, 32});
TestCase(r, "Transparency-affecting composition fills output (ATB,reg)")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-8, -8, 32, 32});
TestCase(r, "Transparency-affecting composition fills output (reg,ATB)")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-8, -8, 32, 32});
}
DEF_TEST_SUITE(TransformedColorFilter, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "Transform composes with regular CF")
.source({0, 0, 24, 24})
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Regular CF composes with transform")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Transform composes with transparency-affecting CF")
.source({0, 0, 24, 24})
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
// NOTE: Because there is no explicit crop between the color filter and the transform,
// output bounds propagation means the layer bounds of the applied color filter are never
// visible post transform. This is detected and allows the transform to be composed without
// producing an intermediate image. See later tests for when a crop prevents this optimization.
TestCase(r, "Transparency-affecting CF composes with transform")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{-50, -50, 50, 50});
}
DEF_TEST_SUITE(TransformBetweenColorFilters, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
// NOTE: The lack of explicit crops allows all of these operations to be optimized as well.
TestCase(r, "Transform between regular color filters")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.75f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Transform between transparency-affecting color filters")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Transform between ATB and regular color filters")
.source({0, 0, 24, 24})
.applyColorFilter(affect_transparent(SkColors::kBlue), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.75f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Transform between regular and ATB color filters")
.source({0, 0, 24, 24})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(45.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
}
DEF_TEST_SUITE(ColorFilterBetweenTransforms, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "Regular color filter between transforms")
.source({0, 0, 24, 24})
.applyTransform(SkMatrix::RotateDeg(20.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.8f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(10.f, {5.f, 8.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
TestCase(r, "Transparency-affecting color filter between transforms")
.source({0, 0, 24, 24})
.applyTransform(SkMatrix::RotateDeg(20.f, {12, 12}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(10.f, {5.f, 8.f}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 24, 24});
}
DEF_TEST_SUITE(CroppedColorFilter, r, CtsEnforcement::kApiLevel_T, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "Regular color filter after empty crop stays empty")
.source({0, 0, 16, 16})
.applyCrop(SkIRect::MakeEmpty(), tm, Expect::kEmptyImage)
.applyColorFilter(alpha_modulate(0.2f), Expect::kEmptyImage)
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "Transparency-affecting color filter after empty crop creates new image")
.source({0, 0, 16, 16})
.applyCrop(SkIRect::MakeEmpty(), tm, Expect::kEmptyImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kNewImage,
/*expectedColorFilter=*/nullptr) // CF applied ASAP to new img
.run(/*requestedOutput=*/{0, 0, 16, 16});
TestCase(r, "Regular color filter composes with crop")
.source({0, 0, 32, 32})
.applyColorFilter(alpha_modulate(0.7f), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop composes with regular color filter")
.source({0, 0, 32, 32})
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
// FIXME need to disable the stats tracking for renderExpected() and compare()
TestCase(r, "Transparency-affecting color filter restricted by crop")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop composes with transparency-affecting color filter")
.source({0, 0, 32, 32})
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
}
}
DEF_TEST_SUITE(CropBetweenColorFilters, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "Crop between regular color filters")
.source({0, 0, 32, 32})
.applyColorFilter(alpha_modulate(0.8f), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.4f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
if (tm == SkTileMode::kDecal) {
TestCase(r, "Crop between transparency-affecting color filters requires new image")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Output-constrained crop between transparency-affecting filters does not")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, SkTileMode::kDecal, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.run(/*requestedOutput=*/{8, 8, 24, 24});
} else {
TestCase(r, "Tiling between transparency-affecting color filters defers image")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
}
TestCase(r, "Crop between regular and ATB color filters")
.source({0, 0, 32, 32})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop between ATB and regular color filters")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kRed), Expect::kDeferredImage)
.applyCrop({8, 8, 24, 24}, tm, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
}
}
DEF_TEST_SUITE(ColorFilterBetweenCrops, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
for (SkTileMode firstTM : kTileModes) {
for (SkTileMode secondTM : kTileModes) {
Expect newImageIfNotDecalOrDoubleClamp =
secondTM != SkTileMode::kDecal &&
!(secondTM == SkTileMode::kClamp && firstTM == SkTileMode::kClamp) ?
Expect::kNewImage : Expect::kDeferredImage;
TestCase(r, "Regular color filter between crops")
.source({0, 0, 32, 32})
.applyCrop({4, 4, 24, 24}, firstTM, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyCrop({15, 15, 32, 32}, secondTM, newImageIfNotDecalOrDoubleClamp,
secondTM == SkTileMode::kDecal ? firstTM : secondTM)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transparency-affecting color filter between crops")
.source({0, 0, 32, 32})
.applyCrop({4, 4, 24, 24}, firstTM, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({15, 15, 32, 32}, secondTM, newImageIfNotDecalOrDoubleClamp,
secondTM == SkTileMode::kDecal ? firstTM : secondTM)
.run(/*requestedOutput=*/{0, 0, 32, 32});
}
}
}
DEF_TEST_SUITE(CroppedTransformedColorFilter, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
TestCase(r, "Transform -> crop -> regular color filter")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transform -> regular color filter -> crop")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop -> transform -> regular color filter")
.source({0, 0, 32, 32})
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop -> regular color filter -> transform")
.source({0, 0, 32, 32})
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Regular color filter -> transform -> crop")
.source({0, 0, 32, 32})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Regular color filter -> crop -> transform")
.source({0, 0, 32, 32})
.applyColorFilter(alpha_modulate(0.5f), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
}
DEF_TEST_SUITE(CroppedTransformedTransparencyAffectingColorFilter, r, CtsEnforcement::kApiLevel_T,
CtsEnforcement::kNextRelease) {
// When the crop is not between the transform and transparency-affecting color filter,
// either the order of operations or the bounds propagation means that every action can be
// deferred. Below, when the crop is between the two actions, new images are triggered.
TestCase(r, "Transform -> transparency-affecting color filter -> crop")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop -> transform -> transparency-affecting color filter")
.source({0, 0, 32, 32})
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Crop -> transparency-affecting color filter -> transform")
.source({0, 0, 32, 32})
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transparency-affecting color filter -> transform -> crop")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
// Since the crop is between the transform and color filter (or vice versa), transparency
// outside the crop is introduced that should not be affected by the color filter were no
// new image to be created.
TestCase(r, "Transform -> crop -> transparency-affecting color filter")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
TestCase(r, "Transparency-affecting color filter -> crop -> transform")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kNewImage)
.run(/*requestedOutput=*/{0, 0, 32, 32});
// However if the output is small enough to fit within the transformed interior, the
// transparency is not visible.
TestCase(r, "Transform -> crop -> transparency-affecting color filter")
.source({0, 0, 32, 32})
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.run(/*requestedOutput=*/{15, 15, 21, 21});
TestCase(r, "Transparency-affecting color filter -> crop -> transform")
.source({0, 0, 32, 32})
.applyColorFilter(affect_transparent(SkColors::kGreen), Expect::kDeferredImage)
.applyCrop({2, 2, 30, 30}, Expect::kDeferredImage)
.applyTransform(SkMatrix::RotateDeg(30.f, {16, 16}), Expect::kDeferredImage)
.run(/*requestedOutput=*/{15, 15, 21, 21});
}
DEF_TEST_SUITE(BackdropFilterRotated, r,
CtsEnforcement::kNextRelease, CtsEnforcement::kNextRelease) {
// These values are extracted from a cc_unittest that had a 200x200 image, with a 10-degree
// rotated 100x200 layer over the right half of the base image, with a backdrop blur. The
// rotation forces SkCanvas to crop and transform the base device's content to be aligned with
// the layer space of the blur. The rotation is such that the backdrop image must be clamped
// (hence the first crop) and the clamp tiling remains visible in the layer image. However,
// floating point precision in the layer bounds analysis was causing FilterResult to think that
// the layer decal was also visible so the first crop would be resolved before the transform was
// applied.
//
// While it's expected that the second clamping crop (part of the blur effect), forces the
// transform and first clamp to be resolved, we were incorrectly producing two new images
// instead of just one.
TestCase(r, "Layer decal shouldn't be visible")
.source({65, 0, 199, 200})
.applyCrop({65, 0, 199, 200}, SkTileMode::kClamp, Expect::kDeferredImage)
.applyTransform(SkMatrix::MakeAll( 0.984808f, 0.173648f, -98.4808f,
-0.173648f, 0.984808f, 17.3648f,
0.000000f, 0.000000f, 1.0000f),
Expect::kDeferredImage)
.applyCrop({0, 0, 100, 200}, SkTileMode::kClamp, Expect::kNewImage)
.run(/*requestedOutput=*/{-15, -15, 115, 215});
}
// Nearly identity rescales are treated as the identity
static constexpr SkSize kNearlyIdentity = {0.999f, 0.999f};
DEF_TEST_SUITE(RescaleWithTileMode, r,
CtsEnforcement::kNextRelease, CtsEnforcement::kNextRelease) {
for (SkTileMode tm : kTileModes) {
TestCase(r, "Identity rescale is a no-op")
.source({0, 0, 50, 50})
.applyCrop({0, 0, 50, 50}, tm, Expect::kDeferredImage)
.rescale({1.f, 1.f}, Expect::kDeferredImage)
.run(/*requestedOutput=*/{-5, -5, 55, 55});
TestCase(r, "Near identity rescale is a no-op",
kDefaultMaxAllowedPercentImageDiff,
/*transparentCheckBorderTolerance=*/tm == SkTileMode::kDecal ? 1 : 0)
.source({0, 0, 50, 50})
.applyCrop({0, 0, 50, 50}, tm, Expect::kDeferredImage)
.rescale(kNearlyIdentity, Expect::kDeferredImage)
.run(/*requestedOutput=*/{-5, -5, 55, 55});
// NOTE: As the scale factor decreases and more decimation steps are required, the testing
// allowed tolerances increase greatly. These were chosen as "acceptable" after reviewing
// the expected vs. actual images. The results diverge due to differences in the simple
// expected decimation and the actual rescale() implementation, as well as how small the
// final images become.
//
// Similarly, the allowed transparent border tolerance must be increased for kDecal tests
// because the expected image's content is expanded by a larger and larger factor during its
// upscale.
// kDecal tiling is applied during the downsample by writing a transparent buffer, so the
// tile mode can simplify to kClamp (which is more efficient for shader-based tiling).
const SkTileMode expectedTileMode = tm == SkTileMode::kDecal ? SkTileMode::kClamp : tm;
TestCase(r, "1-step rescale preserves tile mode",
kDefaultMaxAllowedPercentImageDiff,
/*transparentCheckBorderTolerance=*/tm == SkTileMode::kDecal ? 1 : 0)
.source({16, 16, 64, 64})
.applyCrop({16, 16, 64, 64}, tm, Expect::kDeferredImage</