mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-01 08:42:13 +00:00
0ebc2df843
Differential Revision: https://phabricator.services.mozilla.com/D164732
170 lines
6.0 KiB
C++
170 lines
6.0 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
// Copyright (c) 2011-2016 Google Inc.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the gfx/skia/LICENSE file.
|
|
|
|
#ifndef MOZILLA_GFX_SKCONVOLVER_H_
|
|
#define MOZILLA_GFX_SKCONVOLVER_H_
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include <cfloat>
|
|
#include <cmath>
|
|
#include <vector>
|
|
|
|
namespace skia {
|
|
|
|
class SkBitmapFilter {
|
|
public:
|
|
explicit SkBitmapFilter(float width) : fWidth(width) {}
|
|
virtual ~SkBitmapFilter() = default;
|
|
|
|
float width() const { return fWidth; }
|
|
virtual float evaluate(float x) const = 0;
|
|
|
|
protected:
|
|
float fWidth;
|
|
};
|
|
|
|
class SkBoxFilter final : public SkBitmapFilter {
|
|
public:
|
|
explicit SkBoxFilter(float width = 0.5f) : SkBitmapFilter(width) {}
|
|
|
|
float evaluate(float x) const override {
|
|
return (x >= -fWidth && x < fWidth) ? 1.0f : 0.0f;
|
|
}
|
|
};
|
|
|
|
class SkLanczosFilter final : public SkBitmapFilter {
|
|
public:
|
|
explicit SkLanczosFilter(float width = 3.0f) : SkBitmapFilter(width) {}
|
|
|
|
float evaluate(float x) const override {
|
|
if (x <= -fWidth || x >= fWidth) {
|
|
return 0.0f; // Outside of the window.
|
|
}
|
|
if (x > -FLT_EPSILON && x < FLT_EPSILON) {
|
|
return 1.0f; // Special case the discontinuity at the origin.
|
|
}
|
|
float xpi = x * float(M_PI);
|
|
return (sinf(xpi) / xpi) * // sinc(x)
|
|
sinf(xpi / fWidth) / (xpi / fWidth); // sinc(x/fWidth)
|
|
}
|
|
};
|
|
|
|
// Represents a filter in one dimension. Each output pixel has one entry in this
|
|
// object for the filter values contributing to it. You build up the filter
|
|
// list by calling AddFilter for each output pixel (in order).
|
|
//
|
|
// We do 2-dimensional convolution by first convolving each row by one
|
|
// SkConvolutionFilter1D, then convolving each column by another one.
|
|
//
|
|
// Entries are stored in ConvolutionFixed point, shifted left by kShiftBits.
|
|
class SkConvolutionFilter1D {
|
|
public:
|
|
using ConvolutionFixed = short;
|
|
|
|
// The number of bits that ConvolutionFixed point values are shifted by.
|
|
enum { kShiftBits = 14 };
|
|
|
|
SkConvolutionFilter1D();
|
|
~SkConvolutionFilter1D();
|
|
|
|
// Convert between floating point and our ConvolutionFixed point
|
|
// representation.
|
|
static ConvolutionFixed ToFixed(float f) {
|
|
return static_cast<ConvolutionFixed>(f * (1 << kShiftBits));
|
|
}
|
|
|
|
// Returns the maximum pixel span of a filter.
|
|
int maxFilter() const { return fMaxFilter; }
|
|
|
|
// Returns the number of filters in this filter. This is the dimension of the
|
|
// output image.
|
|
int numValues() const { return static_cast<int>(fFilters.size()); }
|
|
|
|
void reserveAdditional(int filterCount, int filterValueCount) {
|
|
fFilters.reserve(fFilters.size() + filterCount);
|
|
fFilterValues.reserve(fFilterValues.size() + filterValueCount);
|
|
}
|
|
|
|
// Appends the given list of scaling values for generating a given output
|
|
// pixel. |filterOffset| is the distance from the edge of the image to where
|
|
// the scaling factors start. The scaling factors apply to the source pixels
|
|
// starting from this position, and going for the next |filterLength| pixels.
|
|
//
|
|
// You will probably want to make sure your input is normalized (that is,
|
|
// all entries in |filterValuesg| sub to one) to prevent affecting the overall
|
|
// brighness of the image.
|
|
//
|
|
// The filterLength must be > 0.
|
|
void AddFilter(int filterOffset, const ConvolutionFixed* filterValues,
|
|
int filterLength);
|
|
|
|
// Retrieves a filter for the given |valueOffset|, a position in the output
|
|
// image in the direction we're convolving. The offset and length of the
|
|
// filter values are put into the corresponding out arguments (see AddFilter
|
|
// above for what these mean), and a pointer to the first scaling factor is
|
|
// returned. There will be |filterLength| values in this array.
|
|
inline const ConvolutionFixed* FilterForValue(int valueOffset,
|
|
int* filterOffset,
|
|
int* filterLength) const {
|
|
const FilterInstance& filter = fFilters[valueOffset];
|
|
*filterOffset = filter.fOffset;
|
|
*filterLength = filter.fTrimmedLength;
|
|
if (filter.fTrimmedLength == 0) {
|
|
return nullptr;
|
|
}
|
|
return &fFilterValues[filter.fDataLocation];
|
|
}
|
|
|
|
bool ComputeFilterValues(const SkBitmapFilter& aBitmapFilter,
|
|
int32_t aSrcSize, int32_t aDstSize);
|
|
|
|
private:
|
|
struct FilterInstance {
|
|
// Offset within filterValues for this instance of the filter.
|
|
int fDataLocation;
|
|
|
|
// Distance from the left of the filter to the center. IN PIXELS
|
|
int fOffset;
|
|
|
|
// Number of values in this filter instance.
|
|
int fTrimmedLength;
|
|
|
|
// Filter length as specified. Note that this may be different from
|
|
// 'trimmed_length' if leading/trailing zeros of the original floating
|
|
// point form were clipped differently on each tail.
|
|
int fLength;
|
|
};
|
|
|
|
// Stores the information for each filter added to this class.
|
|
std::vector<FilterInstance> fFilters;
|
|
|
|
// We store all the filter values in this flat list, indexed by
|
|
// |FilterInstance.data_location| to avoid the mallocs required for storing
|
|
// each one separately.
|
|
std::vector<ConvolutionFixed> fFilterValues;
|
|
|
|
// The maximum size of any filter we've added.
|
|
int fMaxFilter;
|
|
};
|
|
|
|
void convolve_horizontally(const unsigned char* srcData,
|
|
const SkConvolutionFilter1D& filter,
|
|
unsigned char* outRow, bool hasAlpha);
|
|
|
|
void convolve_vertically(
|
|
const SkConvolutionFilter1D::ConvolutionFixed* filterValues,
|
|
int filterLength, unsigned char* const* sourceDataRows, int pixelWidth,
|
|
unsigned char* outRow, bool hasAlpha);
|
|
|
|
bool BGRAConvolve2D(const unsigned char* sourceData, int sourceByteRowStride,
|
|
bool sourceHasAlpha, const SkConvolutionFilter1D& filterX,
|
|
const SkConvolutionFilter1D& filterY,
|
|
int outputByteRowStride, unsigned char* output);
|
|
|
|
} // namespace skia
|
|
|
|
#endif /* MOZILLA_GFX_SKCONVOLVER_H_ */
|