| 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, | 
|  |