| Index: src/effects/SkBlurMaskFilter.cpp
|
| diff --git a/src/effects/SkBlurMaskFilter.cpp b/src/effects/SkBlurMaskFilter.cpp
|
| index 14be6a50b3d33c8aa8231367b33d4ae9fd52f611..3325232f8b5cfdb00f96b554f37cdcc062e84184 100644
|
| --- a/src/effects/SkBlurMaskFilter.cpp
|
| +++ b/src/effects/SkBlurMaskFilter.cpp
|
| @@ -19,7 +19,10 @@
|
| #if SK_SUPPORT_GPU
|
| #include "GrContext.h"
|
| #include "GrTexture.h"
|
| +#include "GrEffect.h"
|
| +#include "gl/GrGLEffect.h"
|
| #include "effects/GrSimpleTextureEffect.h"
|
| +#include "GrTBackendEffectFactory.h"
|
| #include "SkGrPixelRef.h"
|
| #endif
|
|
|
| @@ -37,6 +40,11 @@ public:
|
| const SkIRect& clipBounds,
|
| const SkMatrix& ctm,
|
| SkRect* maskRect) const SK_OVERRIDE;
|
| + virtual bool directFilterMaskGPU(GrContext* context,
|
| + GrPaint* grp,
|
| + const SkStrokeRec& strokeRec,
|
| + const SkPath& path) const SK_OVERRIDE;
|
| +
|
| virtual bool filterMaskGPU(GrTexture* src,
|
| const SkMatrix& ctm,
|
| const SkRect& maskRect,
|
| @@ -500,6 +508,276 @@ void SkBlurMaskFilterImpl::flatten(SkFlattenableWriteBuffer& buffer) const {
|
|
|
| #if SK_SUPPORT_GPU
|
|
|
| +class GrGLRectBlurEffect;
|
| +
|
| +class GrRectBlurEffect : public GrEffect {
|
| +public:
|
| + virtual ~GrRectBlurEffect();
|
| +
|
| + static const char* Name() { return "RectBlur"; }
|
| +
|
| + typedef GrGLRectBlurEffect GLEffect;
|
| +
|
| + virtual const GrBackendEffectFactory& getFactory() const SK_OVERRIDE;
|
| + virtual void getConstantColorComponents(GrColor* color, uint32_t* validFlags) const SK_OVERRIDE;
|
| +
|
| + /**
|
| + * Create a simple filter effect with custom bicubic coefficients.
|
| + */
|
| + static GrEffectRef* Create(GrContext *context, const SkRect& rect,
|
| + float sigma) {
|
| + GrTexture *horizontalScanline, *verticalScanline;
|
| + bool createdScanlines = CreateScanlineTextures(context, sigma,
|
| + SkScalarCeilToInt(rect.width()),
|
| + SkScalarCeilToInt(rect.height()),
|
| + &horizontalScanline, &verticalScanline);
|
| + if (!createdScanlines) {
|
| + return NULL;
|
| + }
|
| + AutoEffectUnref effect(SkNEW_ARGS(GrRectBlurEffect, (rect, sigma,
|
| + horizontalScanline, verticalScanline)));
|
| + return CreateEffectRef(effect);
|
| + }
|
| +
|
| + unsigned int getWidth() const { return fWidth; }
|
| + unsigned int getHeight() const { return fHeight; }
|
| + float getSigma() const { return fSigma; }
|
| + const GrCoordTransform& getTransform() const { return fTransform; }
|
| +
|
| +private:
|
| + GrRectBlurEffect(const SkRect& rect, float sigma,
|
| + GrTexture *horizontal_scanline, GrTexture *vertical_scanline);
|
| + virtual bool onIsEqual(const GrEffect&) const SK_OVERRIDE;
|
| +
|
| + static bool CreateScanlineTextures(GrContext *context, float sigma,
|
| + unsigned int width, unsigned int height,
|
| + GrTexture **horizontalScanline,
|
| + GrTexture **verticalScanline);
|
| +
|
| + unsigned int fWidth, fHeight;
|
| + float fSigma;
|
| + GrTextureAccess fHorizontalScanlineAccess;
|
| + GrTextureAccess fVerticalScanlineAccess;
|
| + GrCoordTransform fTransform;
|
| +
|
| + GR_DECLARE_EFFECT_TEST;
|
| +
|
| + typedef GrEffect INHERITED;
|
| +};
|
| +
|
| +class GrGLRectBlurEffect : public GrGLEffect {
|
| +public:
|
| + GrGLRectBlurEffect(const GrBackendEffectFactory& factory,
|
| + const GrDrawEffect&);
|
| + virtual void emitCode(GrGLShaderBuilder*,
|
| + const GrDrawEffect&,
|
| + EffectKey,
|
| + const char* outputColor,
|
| + const char* inputColor,
|
| + const TransformedCoordsArray&,
|
| + const TextureSamplerArray&) SK_OVERRIDE;
|
| +
|
| + virtual void setData(const GrGLUniformManager&, const GrDrawEffect&) SK_OVERRIDE;
|
| +
|
| +private:
|
| + typedef GrGLUniformManager::UniformHandle UniformHandle;
|
| +
|
| + UniformHandle fWidthUni;
|
| + UniformHandle fHeightUni;
|
| +
|
| + typedef GrGLEffect INHERITED;
|
| +};
|
| +
|
| +GrGLRectBlurEffect::GrGLRectBlurEffect(const GrBackendEffectFactory& factory, const GrDrawEffect&)
|
| + : INHERITED(factory) {
|
| +}
|
| +
|
| +void GrGLRectBlurEffect::emitCode(GrGLShaderBuilder* builder,
|
| + const GrDrawEffect&,
|
| + EffectKey key,
|
| + const char* outputColor,
|
| + const char* inputColor,
|
| + const TransformedCoordsArray& coords,
|
| + const TextureSamplerArray& samplers) {
|
| +
|
| + SkString texture_coords = builder->ensureFSCoords2D(coords, 0);
|
| +
|
| + if (inputColor) {
|
| + builder->fsCodeAppendf("\tvec4 src=%s;\n", inputColor);
|
| + } else {
|
| + builder->fsCodeAppendf("\tvec4 src=vec4(1)\n;");
|
| + }
|
| +
|
| + builder->fsCodeAppendf("\tvec4 horiz = ");
|
| + builder->fsAppendTextureLookup( samplers[0], texture_coords.c_str() );
|
| + builder->fsCodeAppendf(";\n");
|
| + builder->fsCodeAppendf("\tvec4 vert = ");
|
| + builder->fsAppendTextureLookup( samplers[1], texture_coords.c_str() );
|
| + builder->fsCodeAppendf(";\n");
|
| +
|
| + builder->fsCodeAppendf("\tfloat final = (horiz*vert).r;\n");
|
| + builder->fsCodeAppendf("\t%s = final*src;\n", outputColor);
|
| +}
|
| +
|
| +void GrGLRectBlurEffect::setData(const GrGLUniformManager& uman,
|
| + const GrDrawEffect& drawEffect) {
|
| +}
|
| +
|
| +bool GrRectBlurEffect::CreateScanlineTextures(GrContext *context, float sigma,
|
| + unsigned int width, unsigned int height,
|
| + GrTexture **horizontalScanline,
|
| + GrTexture **verticalScanline) {
|
| + GrTextureParams params;
|
| + GrTextureDesc texDesc;
|
| +
|
| + unsigned int profile_size = SkScalarFloorToInt(6*sigma);
|
| +
|
| + texDesc.fWidth = width;
|
| + texDesc.fHeight = 1;
|
| + texDesc.fConfig = kAlpha_8_GrPixelConfig;
|
| +
|
| + static const GrCacheID::Domain gBlurProfileDomain = GrCacheID::GenerateDomain();
|
| + GrCacheID::Key key;
|
| + memset(&key, 0, sizeof(key));
|
| + key.fData32[0] = profile_size;
|
| + key.fData32[1] = width;
|
| + key.fData32[2] = 1;
|
| + GrCacheID horizontalCacheID(gBlurProfileDomain, key);
|
| +
|
| + uint8_t *profile = NULL;
|
| + SkAutoTDeleteArray<uint8_t> ada(profile);
|
| +
|
| + *horizontalScanline = context->findAndRefTexture(texDesc, horizontalCacheID, ¶ms);
|
| +
|
| + if (NULL == *horizontalScanline) {
|
| +
|
| + SkBlurMask::ComputeBlurProfile(sigma, &profile);
|
| +
|
| + SkAutoTMalloc<uint8_t> horizontalPixels(width);
|
| + SkBlurMask::ComputeBlurredScanline(horizontalPixels, profile, width, sigma);
|
| +
|
| + *horizontalScanline = context->createTexture(¶ms, texDesc, horizontalCacheID,
|
| + horizontalPixels, 0);
|
| +
|
| + if (NULL == *horizontalScanline) {
|
| + return false;
|
| + }
|
| + }
|
| +
|
| + texDesc.fWidth = 1;
|
| + texDesc.fHeight = height;
|
| + key.fData32[1] = 1;
|
| + key.fData32[2] = height;
|
| + GrCacheID verticalCacheID(gBlurProfileDomain, key);
|
| +
|
| + *verticalScanline = context->findAndRefTexture(texDesc, verticalCacheID, ¶ms);
|
| + if (NULL == *verticalScanline) {
|
| + if (NULL == profile) {
|
| + SkBlurMask::ComputeBlurProfile(sigma, &profile);
|
| + }
|
| +
|
| + SkAutoTMalloc<uint8_t> verticalPixels(height);
|
| + SkBlurMask::ComputeBlurredScanline(verticalPixels, profile, height, sigma);
|
| +
|
| + *verticalScanline = context->createTexture(¶ms, texDesc, verticalCacheID,
|
| + verticalPixels, 0);
|
| +
|
| + if (NULL == *verticalScanline) {
|
| + return false;
|
| + }
|
| +
|
| + }
|
| + return true;
|
| +}
|
| +
|
| +GrRectBlurEffect::GrRectBlurEffect(const SkRect& rect, float sigma,
|
| + GrTexture *horizontal_scanline, GrTexture *vertical_scanline)
|
| + : INHERITED(),
|
| + fWidth(horizontal_scanline->width()),
|
| + fHeight(vertical_scanline->width()),
|
| + fSigma(sigma),
|
| + fHorizontalScanlineAccess(horizontal_scanline),
|
| + fVerticalScanlineAccess(vertical_scanline) {
|
| + SkMatrix mat;
|
| + mat.setRectToRect(rect, SkRect::MakeWH(1,1), SkMatrix::kFill_ScaleToFit);
|
| + fTransform.reset(kLocal_GrCoordSet, mat);
|
| + this->addTextureAccess(&fHorizontalScanlineAccess);
|
| + this->addTextureAccess(&fVerticalScanlineAccess);
|
| + this->addCoordTransform(&fTransform);
|
| +}
|
| +
|
| +GrRectBlurEffect::~GrRectBlurEffect() {
|
| +}
|
| +
|
| +const GrBackendEffectFactory& GrRectBlurEffect::getFactory() const {
|
| + return GrTBackendEffectFactory<GrRectBlurEffect>::getInstance();
|
| +}
|
| +
|
| +bool GrRectBlurEffect::onIsEqual(const GrEffect& sBase) const {
|
| + const GrRectBlurEffect& s = CastEffect<GrRectBlurEffect>(sBase);
|
| + return this->getWidth() == s.getWidth() &&
|
| + this->getHeight() == s.getHeight() &&
|
| + this->getSigma() == s.getSigma() &&
|
| + this->getTransform() == s.getTransform();
|
| +}
|
| +
|
| +void GrRectBlurEffect::getConstantColorComponents(GrColor* color, uint32_t* validFlags) const {
|
| + *validFlags = 0;
|
| + return;
|
| +}
|
| +
|
| +GR_DEFINE_EFFECT_TEST(GrRectBlurEffect);
|
| +
|
| +GrEffectRef* GrRectBlurEffect::TestCreate(SkRandom* random,
|
| + GrContext* context,
|
| + const GrDrawTargetCaps&,
|
| + GrTexture**) {
|
| + float sigma = random->nextRangeF(3,8);
|
| + float width = random->nextRangeF(200,300);
|
| + float height = random->nextRangeF(200,300);
|
| + return GrRectBlurEffect::Create(context, SkRect::MakeWH(width, height), sigma);
|
| +}
|
| +
|
| +
|
| +bool SkBlurMaskFilterImpl::directFilterMaskGPU(GrContext* context,
|
| + GrPaint* grp,
|
| + const SkStrokeRec& strokeRec,
|
| + const SkPath& path) const {
|
| + if (fBlurStyle != SkBlurMaskFilter::kNormal_BlurStyle) {
|
| + return false;
|
| + }
|
| +
|
| + SkRect rect;
|
| + if (!path.isRect(&rect)) {
|
| + return false;
|
| + }
|
| +
|
| + if (!strokeRec.isFillStyle()) {
|
| + return false;
|
| + }
|
| +
|
| + SkMatrix ctm = context->getMatrix();
|
| + SkScalar xformedSigma = this->computeXformedSigma(ctm);
|
| + rect.outset(3*xformedSigma, 3*xformedSigma);
|
| +
|
| + SkAutoTUnref<GrEffectRef> effect(GrRectBlurEffect::Create(
|
| + context, rect, xformedSigma));
|
| + if (!effect) {
|
| + return false;
|
| + }
|
| +
|
| + GrContext::AutoMatrix am;
|
| + if (!am.setIdentity(context, grp)) {
|
| + return false;
|
| + }
|
| +
|
| +
|
| + grp->addCoverageEffect(effect);
|
| +
|
| + context->drawRect(*grp, rect);
|
| + return true;
|
| +}
|
| +
|
| bool SkBlurMaskFilterImpl::canFilterMaskGPU(const SkRect& srcBounds,
|
| const SkIRect& clipBounds,
|
| const SkMatrix& ctm,
|
|
|