Bin circular blur profile textures by scale and blur to radius ratio.
This reduces the number of profile textures that will be generated throughout an animation and also caps the texture size. It could probably be better tuned. GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=2062743003 Review-Url: https://codereview.chromium.org/2062743003
This commit is contained in:
parent
59d6402ba7
commit
f7fcdb226d
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#include "SkFixed.h"
|
#include "SkFixed.h"
|
||||||
|
|
||||||
class GrGLCircleBlurFragmentProcessor : public GrGLSLFragmentProcessor {
|
class GrCircleBlurFragmentProcessor::GLSLProcessor : public GrGLSLFragmentProcessor {
|
||||||
public:
|
public:
|
||||||
void emitCode(EmitArgs&) override;
|
void emitCode(EmitArgs&) override;
|
||||||
|
|
||||||
@ -33,14 +33,14 @@ private:
|
|||||||
typedef GrGLSLFragmentProcessor INHERITED;
|
typedef GrGLSLFragmentProcessor INHERITED;
|
||||||
};
|
};
|
||||||
|
|
||||||
void GrGLCircleBlurFragmentProcessor::emitCode(EmitArgs& args) {
|
void GrCircleBlurFragmentProcessor::GLSLProcessor::emitCode(EmitArgs& args) {
|
||||||
|
|
||||||
const char *dataName;
|
const char *dataName;
|
||||||
|
|
||||||
// The data is formatted as:
|
// The data is formatted as:
|
||||||
// x,y - the center of the circle
|
// x,y - the center of the circle
|
||||||
// z - the distance at which the intensity starts falling off (e.g., the start of the table)
|
// z - the distance at which the intensity starts falling off (e.g., the start of the table)
|
||||||
// w - the inverse of the profile texture size
|
// w - the inverse of the distance over which the texture is stretched.
|
||||||
fDataUniform = args.fUniformHandler->addUniform(kFragment_GrShaderFlag,
|
fDataUniform = args.fUniformHandler->addUniform(kFragment_GrShaderFlag,
|
||||||
kVec4f_GrSLType,
|
kVec4f_GrSLType,
|
||||||
kDefault_GrSLPrecision,
|
kDefault_GrSLPrecision,
|
||||||
@ -71,28 +71,30 @@ void GrGLCircleBlurFragmentProcessor::emitCode(EmitArgs& args) {
|
|||||||
fragBuilder->codeAppendf("%s = src * intensity;\n", args.fOutputColor );
|
fragBuilder->codeAppendf("%s = src * intensity;\n", args.fOutputColor );
|
||||||
}
|
}
|
||||||
|
|
||||||
void GrGLCircleBlurFragmentProcessor::onSetData(const GrGLSLProgramDataManager& pdman,
|
void GrCircleBlurFragmentProcessor::GLSLProcessor::onSetData(const GrGLSLProgramDataManager& pdman,
|
||||||
const GrProcessor& proc) {
|
const GrProcessor& proc) {
|
||||||
const GrCircleBlurFragmentProcessor& cbfp = proc.cast<GrCircleBlurFragmentProcessor>();
|
const GrCircleBlurFragmentProcessor& cbfp = proc.cast<GrCircleBlurFragmentProcessor>();
|
||||||
const SkRect& circle = cbfp.circle();
|
const SkRect& circle = cbfp.fCircle;
|
||||||
|
|
||||||
// The data is formatted as:
|
// The data is formatted as:
|
||||||
// x,y - the center of the circle
|
// x,y - the center of the circle
|
||||||
// z - the distance at which the intensity starts falling off (e.g., the start of the table)
|
// z - the distance at which the intensity starts falling off (e.g., the start of the table)
|
||||||
// w - the inverse of the profile texture size
|
// w - the inverse of the distance over which the profile texture is stretched.
|
||||||
pdman.set4f(fDataUniform, circle.centerX(), circle.centerY(), cbfp.offset(),
|
pdman.set4f(fDataUniform, circle.centerX(), circle.centerY(), cbfp.fSolidRadius,
|
||||||
1.0f / cbfp.profileSize());
|
1.f / cbfp.fTextureRadius);
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
GrCircleBlurFragmentProcessor::GrCircleBlurFragmentProcessor(const SkRect& circle,
|
GrCircleBlurFragmentProcessor::GrCircleBlurFragmentProcessor(const SkRect& circle,
|
||||||
float sigma,
|
float sigma,
|
||||||
float offset,
|
float solidRadius,
|
||||||
|
float textureRadius,
|
||||||
GrTexture* blurProfile)
|
GrTexture* blurProfile)
|
||||||
: fCircle(circle)
|
: fCircle(circle)
|
||||||
, fSigma(sigma)
|
, fSigma(sigma)
|
||||||
, fOffset(offset)
|
, fSolidRadius(solidRadius)
|
||||||
|
, fTextureRadius(textureRadius)
|
||||||
, fBlurProfileAccess(blurProfile, GrTextureParams::kBilerp_FilterMode) {
|
, fBlurProfileAccess(blurProfile, GrTextureParams::kBilerp_FilterMode) {
|
||||||
this->initClassID<GrCircleBlurFragmentProcessor>();
|
this->initClassID<GrCircleBlurFragmentProcessor>();
|
||||||
this->addTextureAccess(&fBlurProfileAccess);
|
this->addTextureAccess(&fBlurProfileAccess);
|
||||||
@ -100,12 +102,13 @@ GrCircleBlurFragmentProcessor::GrCircleBlurFragmentProcessor(const SkRect& circl
|
|||||||
}
|
}
|
||||||
|
|
||||||
GrGLSLFragmentProcessor* GrCircleBlurFragmentProcessor::onCreateGLSLInstance() const {
|
GrGLSLFragmentProcessor* GrCircleBlurFragmentProcessor::onCreateGLSLInstance() const {
|
||||||
return new GrGLCircleBlurFragmentProcessor;
|
return new GLSLProcessor;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GrCircleBlurFragmentProcessor::onGetGLSLProcessorKey(const GrGLSLCaps& caps,
|
void GrCircleBlurFragmentProcessor::onGetGLSLProcessorKey(const GrGLSLCaps& caps,
|
||||||
GrProcessorKeyBuilder* b) const {
|
GrProcessorKeyBuilder* b) const {
|
||||||
GrGLCircleBlurFragmentProcessor::GenKey(*this, caps, b);
|
// The code for this processor is always the same so there is nothing to add to the key.
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GrCircleBlurFragmentProcessor::onComputeInvariantOutput(GrInvariantOutput* inout) const {
|
void GrCircleBlurFragmentProcessor::onComputeInvariantOutput(GrInvariantOutput* inout) const {
|
||||||
@ -193,21 +196,6 @@ static uint8_t eval_at(float evalX, float circleR, const float* halfKernel, int
|
|||||||
return SkUnitScalarClampToByte(2.f * acc);
|
return SkUnitScalarClampToByte(2.f * acc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void compute_profile_offset_and_size(float circleR, float sigma,
|
|
||||||
float* offset, int* size) {
|
|
||||||
if (3*sigma <= circleR) {
|
|
||||||
// The circle is bigger than the Gaussian. In this case we know the interior of the
|
|
||||||
// blurred circle is solid.
|
|
||||||
*offset = circleR - 3 * sigma; // This location maps to 0.5f in the weights texture.
|
|
||||||
// It should always be 255.
|
|
||||||
*size = SkScalarCeilToInt(6*sigma);
|
|
||||||
} else {
|
|
||||||
// The Gaussian is bigger than the circle.
|
|
||||||
*offset = 0.0f;
|
|
||||||
*size = SkScalarCeilToInt(circleR + 3*sigma);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function creates a profile of a blurred circle. It does this by computing a kernel for
|
// This function creates a profile of a blurred circle. It does this by computing a kernel for
|
||||||
// half the Gaussian and a matching summed area table. The summed area table is used to compute
|
// half the Gaussian and a matching summed area table. The summed area table is used to compute
|
||||||
// an array of vertical applications of the half kernel to the circle along the x axis. The table
|
// an array of vertical applications of the half kernel to the circle along the x axis. The table
|
||||||
@ -215,11 +203,8 @@ static inline void compute_profile_offset_and_size(float circleR, float sigma,
|
|||||||
// of the profile being computed. Then for each of the n profile entries we walk out k steps in each
|
// of the profile being computed. Then for each of the n profile entries we walk out k steps in each
|
||||||
// horizontal direction multiplying the corresponding y evaluation by the half kernel entry and
|
// horizontal direction multiplying the corresponding y evaluation by the half kernel entry and
|
||||||
// sum these values to compute the profile entry.
|
// sum these values to compute the profile entry.
|
||||||
static uint8_t* create_profile(float circleR, float sigma) {
|
static uint8_t* create_profile(float sigma, float circleR, float offset, int profileTextureWidth) {
|
||||||
float offset;
|
const int numSteps = profileTextureWidth;
|
||||||
int numSteps;
|
|
||||||
compute_profile_offset_and_size(circleR, sigma, &offset, &numSteps);
|
|
||||||
|
|
||||||
uint8_t* weights = new uint8_t[numSteps];
|
uint8_t* weights = new uint8_t[numSteps];
|
||||||
|
|
||||||
// The full kernel is 6 sigmas wide.
|
// The full kernel is 6 sigmas wide.
|
||||||
@ -248,35 +233,75 @@ static uint8_t* create_profile(float circleR, float sigma) {
|
|||||||
return weights;
|
return weights;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int next_pow2_16bits(int x) {
|
||||||
|
SkASSERT(x > 0);
|
||||||
|
SkASSERT(x <= SK_MaxS16);
|
||||||
|
x--;
|
||||||
|
x |= x >> 1;
|
||||||
|
x |= x >> 2;
|
||||||
|
x |= x >> 4;
|
||||||
|
x |= x >> 8;
|
||||||
|
return x + 1;
|
||||||
|
}
|
||||||
|
|
||||||
GrTexture* GrCircleBlurFragmentProcessor::CreateCircleBlurProfileTexture(
|
GrTexture* GrCircleBlurFragmentProcessor::CreateCircleBlurProfileTexture(
|
||||||
GrTextureProvider* textureProvider,
|
GrTextureProvider* textureProvider,
|
||||||
const SkRect& circle,
|
const SkRect& circle,
|
||||||
float sigma,
|
float sigma,
|
||||||
float* offset) {
|
float* solidRadius,
|
||||||
|
float* textureRadius) {
|
||||||
float circleR = circle.width() / 2.0f;
|
float circleR = circle.width() / 2.0f;
|
||||||
|
// Profile textures are cached by the ratio of sigma to circle radius and by the size of the
|
||||||
|
// profile texture (binned by powers of 2).
|
||||||
|
SkScalar sigmaToCircleRRatio = sigma / circleR;
|
||||||
|
// When sigma is really small this becomes a equivalent to convolving a Gaussian with a half-
|
||||||
|
// plane. We could do that simpler computation. However, right now we're just using a lower
|
||||||
|
// bound off the ratio. Similarly, in the extreme high ratio cases circle becomes a point WRT to
|
||||||
|
// the Guassian and the profile texture is a just a Gaussian evaluation.
|
||||||
|
sigmaToCircleRRatio = SkTPin(sigmaToCircleRRatio, 0.05f, 8.f);
|
||||||
|
// Convert to fixed point for the key.
|
||||||
|
SkFixed sigmaToCircleRRatioFixed = SkScalarToFixed(sigmaToCircleRRatio);
|
||||||
|
// We shave off some bits to reduce the number of unique entries. We could probably shave off
|
||||||
|
// more than we do.
|
||||||
|
sigmaToCircleRRatioFixed &= ~0xff;
|
||||||
|
// From the circle center to solidRadius is all 1s and represented by the leftmost pixel (with
|
||||||
|
// value 255) in the profile texture. If it is zero then there is no solid center to the
|
||||||
|
// blurred circle.
|
||||||
|
if (3*sigma <= circleR) {
|
||||||
|
// The circle is bigger than the Gaussian. In this case we know the interior of the
|
||||||
|
// blurred circle is solid.
|
||||||
|
*solidRadius = circleR - 3 * sigma; // This location maps to 0.5f in the weights texture.
|
||||||
|
// It should always be 255.
|
||||||
|
*textureRadius = SkScalarCeilToScalar(6 * sigma);
|
||||||
|
} else {
|
||||||
|
// The Gaussian is bigger than the circle.
|
||||||
|
*solidRadius = 0.0f;
|
||||||
|
*textureRadius = SkScalarCeilToScalar(circleR + 3 * sigma);
|
||||||
|
}
|
||||||
|
int profileTextureWidth = SkScalarCeilToInt(*textureRadius);
|
||||||
|
profileTextureWidth = (profileTextureWidth >= 1024) ? 1024 :
|
||||||
|
next_pow2_16bits(profileTextureWidth);
|
||||||
|
|
||||||
static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain();
|
static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain();
|
||||||
GrUniqueKey key;
|
GrUniqueKey key;
|
||||||
GrUniqueKey::Builder builder(&key, kDomain, 2);
|
GrUniqueKey::Builder builder(&key, kDomain, 2);
|
||||||
// The profile curve varies with both the sigma of the Gaussian and the size of the
|
builder[0] = sigmaToCircleRRatioFixed;
|
||||||
// disk. Quantizing to 16.16 should be close enough though.
|
builder[1] = profileTextureWidth;
|
||||||
builder[0] = SkScalarToFixed(sigma);
|
|
||||||
builder[1] = SkScalarToFixed(circleR);
|
|
||||||
builder.finish();
|
builder.finish();
|
||||||
|
|
||||||
GrTexture *blurProfile = textureProvider->findAndRefTextureByUniqueKey(key);
|
GrTexture *blurProfile = textureProvider->findAndRefTextureByUniqueKey(key);
|
||||||
|
|
||||||
int profileSize;
|
|
||||||
compute_profile_offset_and_size(circleR, sigma, offset, &profileSize);
|
|
||||||
|
|
||||||
if (!blurProfile) {
|
if (!blurProfile) {
|
||||||
|
|
||||||
GrSurfaceDesc texDesc;
|
GrSurfaceDesc texDesc;
|
||||||
texDesc.fWidth = profileSize;
|
texDesc.fWidth = profileTextureWidth;
|
||||||
texDesc.fHeight = 1;
|
texDesc.fHeight = 1;
|
||||||
texDesc.fConfig = kAlpha_8_GrPixelConfig;
|
texDesc.fConfig = kAlpha_8_GrPixelConfig;
|
||||||
|
|
||||||
SkAutoTDeleteArray<uint8_t> profile(create_profile(circleR, sigma));
|
// Rescale params to the size of the texture we're creating.
|
||||||
|
SkScalar scale = profileTextureWidth / *textureRadius;
|
||||||
|
SkAutoTDeleteArray<uint8_t> profile(create_profile(sigma * scale, circleR * scale,
|
||||||
|
*solidRadius * scale,
|
||||||
|
profileTextureWidth));
|
||||||
|
|
||||||
blurProfile = textureProvider->createTexture(texDesc, SkBudgeted::kYes, profile.get(), 0);
|
blurProfile = textureProvider->createTexture(texDesc, SkBudgeted::kYes, profile.get(), 0);
|
||||||
if (blurProfile) {
|
if (blurProfile) {
|
||||||
|
@ -28,35 +28,41 @@ public:
|
|||||||
|
|
||||||
SkString dumpInfo() const override {
|
SkString dumpInfo() const override {
|
||||||
SkString str;
|
SkString str;
|
||||||
str.appendf("Rect [L: %.2f, T: %.2f, R: %.2f, B: %.2f], Sigma %.2f, Offset: %.2f",
|
str.appendf("Rect [L: %.2f, T: %.2f, R: %.2f, B: %.2f], Sigma %.2f, solidR: %.2f, "
|
||||||
|
"textureR: %.2f",
|
||||||
fCircle.fLeft, fCircle.fTop, fCircle.fRight, fCircle.fBottom,
|
fCircle.fLeft, fCircle.fTop, fCircle.fRight, fCircle.fBottom,
|
||||||
fSigma, fOffset);
|
fSigma, fSolidRadius, fTextureRadius);
|
||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
static sk_sp<GrFragmentProcessor> Make(GrTextureProvider*textureProvider,
|
static sk_sp<GrFragmentProcessor> Make(GrTextureProvider*textureProvider,
|
||||||
const SkRect& circle, float sigma) {
|
const SkRect& circle, float sigma) {
|
||||||
float offset;
|
float solidRadius;
|
||||||
|
float textureRadius;
|
||||||
|
|
||||||
SkAutoTUnref<GrTexture> blurProfile(CreateCircleBlurProfileTexture(textureProvider,
|
SkAutoTUnref<GrTexture> profile(CreateCircleBlurProfileTexture(textureProvider,
|
||||||
circle,
|
circle,
|
||||||
sigma,
|
sigma,
|
||||||
&offset));
|
&solidRadius,
|
||||||
if (!blurProfile) {
|
&textureRadius));
|
||||||
|
if (!profile) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return sk_sp<GrFragmentProcessor>(
|
return sk_sp<GrFragmentProcessor>(
|
||||||
new GrCircleBlurFragmentProcessor(circle, sigma, offset, blurProfile));
|
new GrCircleBlurFragmentProcessor(circle, sigma, solidRadius, textureRadius, profile));
|
||||||
}
|
}
|
||||||
|
|
||||||
const SkRect& circle() const { return fCircle; }
|
|
||||||
float sigma() const { return fSigma; }
|
|
||||||
float offset() const { return fOffset; }
|
|
||||||
int profileSize() const { return fBlurProfileAccess.getTexture()->width(); }
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
// This nested GLSL processor implementation is defined in the cpp file.
|
||||||
|
class GLSLProcessor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a profile texture for the circle and sigma. The texture will have a height of 1.
|
||||||
|
* The x texture coord should map from 0 to 1 across the radius range of solidRadius to
|
||||||
|
* solidRadius + textureRadius.
|
||||||
|
*/
|
||||||
GrCircleBlurFragmentProcessor(const SkRect& circle, float sigma,
|
GrCircleBlurFragmentProcessor(const SkRect& circle, float sigma,
|
||||||
float offset, GrTexture* blurProfile);
|
float solidRadius, float textureRadius, GrTexture* blurProfile);
|
||||||
|
|
||||||
GrGLSLFragmentProcessor* onCreateGLSLInstance() const override;
|
GrGLSLFragmentProcessor* onCreateGLSLInstance() const override;
|
||||||
|
|
||||||
@ -65,18 +71,21 @@ private:
|
|||||||
bool onIsEqual(const GrFragmentProcessor& other) const override {
|
bool onIsEqual(const GrFragmentProcessor& other) const override {
|
||||||
const GrCircleBlurFragmentProcessor& cbfp = other.cast<GrCircleBlurFragmentProcessor>();
|
const GrCircleBlurFragmentProcessor& cbfp = other.cast<GrCircleBlurFragmentProcessor>();
|
||||||
// fOffset is computed from the circle width and the sigma
|
// fOffset is computed from the circle width and the sigma
|
||||||
return this->circle() == cbfp.circle() && fSigma == cbfp.fSigma;
|
return this->fCircle == cbfp.fCircle && fSigma == cbfp.fSigma;
|
||||||
}
|
}
|
||||||
|
|
||||||
void onComputeInvariantOutput(GrInvariantOutput* inout) const override;
|
void onComputeInvariantOutput(GrInvariantOutput* inout) const override;
|
||||||
|
|
||||||
static GrTexture* CreateCircleBlurProfileTexture(GrTextureProvider*,
|
static GrTexture* CreateCircleBlurProfileTexture(GrTextureProvider*,
|
||||||
const SkRect& circle,
|
const SkRect& circle,
|
||||||
float sigma, float* offset);
|
float sigma,
|
||||||
|
float* solidRadius,
|
||||||
|
float* textureRadius);
|
||||||
|
|
||||||
SkRect fCircle;
|
SkRect fCircle;
|
||||||
float fSigma;
|
float fSigma;
|
||||||
float fOffset;
|
float fSolidRadius;
|
||||||
|
float fTextureRadius;
|
||||||
GrTextureAccess fBlurProfileAccess;
|
GrTextureAccess fBlurProfileAccess;
|
||||||
|
|
||||||
GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
|
GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
|
||||||
|
Loading…
Reference in New Issue
Block a user