senorblanco@chromium.org | 60014ca | 2011-11-09 16:05:58 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2011 The Android Open Source Project |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 8 | #include "SkAutoPixmapStorage.h" |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 9 | #include "SkColorPriv.h" |
mtklein | dce5ce4 | 2015-08-04 08:49:21 -0700 | [diff] [blame] | 10 | #include "SkGpuBlurUtils.h" |
| 11 | #include "SkOpts.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 12 | #include "SkReadBuffer.h" |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 13 | #include "SkSpecialImage.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 14 | #include "SkWriteBuffer.h" |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 15 | |
bsalomon@google.com | cf8fb1f | 2012-08-02 14:03:32 +0000 | [diff] [blame] | 16 | #if SK_SUPPORT_GPU |
senorblanco@chromium.org | 302cffb | 2012-08-01 20:16:34 +0000 | [diff] [blame] | 17 | #include "GrContext.h" |
robertphillips | d728f0c | 2016-11-21 11:05:03 -0800 | [diff] [blame] | 18 | #include "GrTextureProxy.h" |
robertphillips | 1de87df | 2016-01-14 06:03:29 -0800 | [diff] [blame] | 19 | #include "SkGr.h" |
bsalomon@google.com | cf8fb1f | 2012-08-02 14:03:32 +0000 | [diff] [blame] | 20 | #endif |
senorblanco@chromium.org | 60014ca | 2011-11-09 16:05:58 +0000 | [diff] [blame] | 21 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 22 | class SkBlurImageFilterImpl : public SkImageFilter { |
| 23 | public: |
| 24 | SkBlurImageFilterImpl(SkScalar sigmaX, |
| 25 | SkScalar sigmaY, |
| 26 | sk_sp<SkImageFilter> input, |
| 27 | const CropRect* cropRect); |
| 28 | |
| 29 | SkRect computeFastBounds(const SkRect&) const override; |
| 30 | |
| 31 | SK_TO_STRING_OVERRIDE() |
| 32 | SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkBlurImageFilterImpl) |
| 33 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 34 | protected: |
| 35 | void flatten(SkWriteBuffer&) const override; |
| 36 | sk_sp<SkSpecialImage> onFilterImage(SkSpecialImage* source, const Context&, |
| 37 | SkIPoint* offset) const override; |
| 38 | SkIRect onFilterNodeBounds(const SkIRect& src, const SkMatrix&, MapDirection) const override; |
| 39 | |
| 40 | private: |
| 41 | SkSize fSigma; |
| 42 | typedef SkImageFilter INHERITED; |
| 43 | |
| 44 | friend class SkImageFilter; |
| 45 | }; |
| 46 | |
| 47 | SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_START(SkImageFilter) |
| 48 | SK_DEFINE_FLATTENABLE_REGISTRAR_ENTRY(SkBlurImageFilterImpl) |
| 49 | SK_DEFINE_FLATTENABLE_REGISTRAR_GROUP_END |
| 50 | |
| 51 | /////////////////////////////////////////////////////////////////////////////// |
| 52 | |
| 53 | sk_sp<SkImageFilter> SkImageFilter::MakeBlur(SkScalar sigmaX, SkScalar sigmaY, |
robertphillips | 225db44 | 2016-04-17 14:27:05 -0700 | [diff] [blame] | 54 | sk_sp<SkImageFilter> input, |
| 55 | const CropRect* cropRect) { |
| 56 | if (0 == sigmaX && 0 == sigmaY && !cropRect) { |
| 57 | return input; |
| 58 | } |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 59 | return sk_sp<SkImageFilter>(new SkBlurImageFilterImpl(sigmaX, sigmaY, input, cropRect)); |
robertphillips | 225db44 | 2016-04-17 14:27:05 -0700 | [diff] [blame] | 60 | } |
| 61 | |
senorblanco@chromium.org | 09843fd | 2014-03-24 20:50:59 +0000 | [diff] [blame] | 62 | // This rather arbitrary-looking value results in a maximum box blur kernel size |
| 63 | // of 1000 pixels on the raster path, which matches the WebKit and Firefox |
| 64 | // implementations. Since the GPU path does not compute a box blur, putting |
| 65 | // the limit on sigma ensures consistent behaviour between the GPU and |
| 66 | // raster paths. |
| 67 | #define MAX_SIGMA SkIntToScalar(532) |
| 68 | |
senorblanco | 6404981 | 2016-02-01 10:32:42 -0800 | [diff] [blame] | 69 | static SkVector map_sigma(const SkSize& localSigma, const SkMatrix& ctm) { |
senorblanco | 32673b9 | 2014-09-09 09:15:04 -0700 | [diff] [blame] | 70 | SkVector sigma = SkVector::Make(localSigma.width(), localSigma.height()); |
| 71 | ctm.mapVectors(&sigma, 1); |
| 72 | sigma.fX = SkMinScalar(SkScalarAbs(sigma.fX), MAX_SIGMA); |
| 73 | sigma.fY = SkMinScalar(SkScalarAbs(sigma.fY), MAX_SIGMA); |
| 74 | return sigma; |
| 75 | } |
| 76 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 77 | SkBlurImageFilterImpl::SkBlurImageFilterImpl(SkScalar sigmaX, |
senorblanco@chromium.org | 194d775 | 2013-07-24 22:19:24 +0000 | [diff] [blame] | 78 | SkScalar sigmaY, |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 79 | sk_sp<SkImageFilter> input, |
senorblanco | 24e06d5 | 2015-03-18 12:11:33 -0700 | [diff] [blame] | 80 | const CropRect* cropRect) |
robertphillips | 6e7025a | 2016-04-04 04:31:25 -0700 | [diff] [blame] | 81 | : INHERITED(&input, 1, cropRect) |
| 82 | , fSigma(SkSize::Make(sigmaX, sigmaY)) { |
senorblanco@chromium.org | 60014ca | 2011-11-09 16:05:58 +0000 | [diff] [blame] | 83 | } |
| 84 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 85 | sk_sp<SkFlattenable> SkBlurImageFilterImpl::CreateProc(SkReadBuffer& buffer) { |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 86 | SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1); |
| 87 | SkScalar sigmaX = buffer.readScalar(); |
| 88 | SkScalar sigmaY = buffer.readScalar(); |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 89 | return SkImageFilter::MakeBlur(sigmaX, sigmaY, common.getInput(0), &common.cropRect()); |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 90 | } |
| 91 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 92 | void SkBlurImageFilterImpl::flatten(SkWriteBuffer& buffer) const { |
senorblanco@chromium.org | 54e01b2 | 2011-11-16 18:20:47 +0000 | [diff] [blame] | 93 | this->INHERITED::flatten(buffer); |
| 94 | buffer.writeScalar(fSigma.fWidth); |
| 95 | buffer.writeScalar(fSigma.fHeight); |
| 96 | } |
| 97 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 98 | static void get_box3_params(SkScalar s, int *kernelSize, int* kernelSize3, int *lowOffset, |
| 99 | int *highOffset) { |
schenney@chromium.org | 73f3ded | 2011-12-20 22:31:40 +0000 | [diff] [blame] | 100 | float pi = SkScalarToFloat(SK_ScalarPI); |
| 101 | int d = static_cast<int>(floorf(SkScalarToFloat(s) * 3.0f * sqrtf(2.0f * pi) / 4.0f + 0.5f)); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 102 | *kernelSize = d; |
| 103 | if (d % 2 == 1) { |
| 104 | *lowOffset = *highOffset = (d - 1) / 2; |
| 105 | *kernelSize3 = d; |
| 106 | } else { |
| 107 | *highOffset = d / 2; |
| 108 | *lowOffset = *highOffset - 1; |
| 109 | *kernelSize3 = d + 1; |
| 110 | } |
| 111 | } |
| 112 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 113 | sk_sp<SkSpecialImage> SkBlurImageFilterImpl::onFilterImage(SkSpecialImage* source, |
robertphillips | d728f0c | 2016-11-21 11:05:03 -0800 | [diff] [blame] | 114 | const Context& ctx, |
| 115 | SkIPoint* offset) const { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 116 | SkIPoint inputOffset = SkIPoint::Make(0, 0); |
| 117 | |
robertphillips | 2302de9 | 2016-03-24 07:26:32 -0700 | [diff] [blame] | 118 | sk_sp<SkSpecialImage> input(this->filterInput(0, source, ctx, &inputOffset)); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 119 | if (!input) { |
| 120 | return nullptr; |
senorblanco@chromium.org | 6840076 | 2013-05-24 15:04:07 +0000 | [diff] [blame] | 121 | } |
| 122 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 123 | SkIRect inputBounds = SkIRect::MakeXYWH(inputOffset.fX, inputOffset.fY, |
| 124 | input->width(), input->height()); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 125 | |
senorblanco | afec27f | 2016-02-16 09:11:18 -0800 | [diff] [blame] | 126 | SkIRect dstBounds; |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 127 | if (!this->applyCropRect(this->mapContext(ctx), inputBounds, &dstBounds)) { |
| 128 | return nullptr; |
senorblanco | db64af3 | 2015-12-09 10:11:43 -0800 | [diff] [blame] | 129 | } |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 130 | if (!inputBounds.intersect(dstBounds)) { |
| 131 | return nullptr; |
reed@google.com | 76dd277 | 2012-01-05 21:15:07 +0000 | [diff] [blame] | 132 | } |
| 133 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 134 | const SkVector sigma = map_sigma(fSigma, ctx.ctm()); |
| 135 | |
| 136 | #if SK_SUPPORT_GPU |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 137 | if (source->isTextureBacked()) { |
| 138 | GrContext* context = source->getContext(); |
Brian Osman | 5f50092 | 2016-12-29 11:50:48 -0500 | [diff] [blame^] | 139 | |
| 140 | // Ensure the input is in the destination's gamut. This saves us from having to do the |
| 141 | // xform during the filter itself. |
| 142 | input = ImageToColorSpace(input.get(), ctx.outputProperties()); |
| 143 | |
robertphillips | c91fd34 | 2016-04-25 12:32:54 -0700 | [diff] [blame] | 144 | sk_sp<GrTexture> inputTexture(input->asTextureRef(context)); |
Robert Phillips | 833dcf4 | 2016-11-18 08:44:13 -0500 | [diff] [blame] | 145 | if (!inputTexture) { |
| 146 | return nullptr; |
| 147 | } |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 148 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 149 | if (0 == sigma.x() && 0 == sigma.y()) { |
| 150 | offset->fX = inputBounds.x(); |
| 151 | offset->fY = inputBounds.y(); |
| 152 | return input->makeSubset(inputBounds.makeOffset(-inputOffset.x(), |
robertphillips | 2302de9 | 2016-03-24 07:26:32 -0700 | [diff] [blame] | 153 | -inputOffset.y())); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 154 | } |
| 155 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 156 | offset->fX = dstBounds.fLeft; |
| 157 | offset->fY = dstBounds.fTop; |
| 158 | inputBounds.offset(-inputOffset); |
| 159 | dstBounds.offset(-inputOffset); |
Brian Osman | 5f50092 | 2016-12-29 11:50:48 -0500 | [diff] [blame^] | 160 | // Typically, we would create the RTC with the output's color space (from ctx), but we |
| 161 | // always blur in the PixelConfig of the *input*. Those might not be compatible (if they |
| 162 | // have different transfer functions). We've already guaranteed that those color spaces |
| 163 | // have the same gamut, so in this case, we do everything in the input's color space. |
Brian Osman | 1105224 | 2016-10-27 14:47:55 -0400 | [diff] [blame] | 164 | sk_sp<GrRenderTargetContext> renderTargetContext(SkGpuBlurUtils::GaussianBlur( |
brianosman | dfe4f2e | 2016-07-21 13:28:36 -0700 | [diff] [blame] | 165 | context, |
| 166 | inputTexture.get(), |
Brian Osman | 5f50092 | 2016-12-29 11:50:48 -0500 | [diff] [blame^] | 167 | sk_ref_sp(input->getColorSpace()), |
brianosman | dfe4f2e | 2016-07-21 13:28:36 -0700 | [diff] [blame] | 168 | dstBounds, |
| 169 | &inputBounds, |
| 170 | sigma.x(), |
| 171 | sigma.y())); |
Brian Osman | 1105224 | 2016-10-27 14:47:55 -0400 | [diff] [blame] | 172 | if (!renderTargetContext) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 173 | return nullptr; |
| 174 | } |
| 175 | |
Robert Phillips | 51d77ff | 2016-12-02 14:56:45 -0500 | [diff] [blame] | 176 | return SkSpecialImage::MakeFromGpu(SkIRect::MakeWH(dstBounds.width(), dstBounds.height()), |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 177 | kNeedNewImageUniqueID_SpecialImage, |
Robert Phillips | 51d77ff | 2016-12-02 14:56:45 -0500 | [diff] [blame] | 178 | renderTargetContext->asTexture(), |
Brian Osman | 5f50092 | 2016-12-29 11:50:48 -0500 | [diff] [blame^] | 179 | sk_ref_sp(renderTargetContext->getColorSpace()), |
| 180 | &source->props()); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 181 | } |
| 182 | #endif |
senorblanco@chromium.org | 2bfe36b | 2014-01-20 19:58:28 +0000 | [diff] [blame] | 183 | |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 184 | int kernelSizeX, kernelSizeX3, lowOffsetX, highOffsetX; |
| 185 | int kernelSizeY, kernelSizeY3, lowOffsetY, highOffsetY; |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 186 | get_box3_params(sigma.x(), &kernelSizeX, &kernelSizeX3, &lowOffsetX, &highOffsetX); |
| 187 | get_box3_params(sigma.y(), &kernelSizeY, &kernelSizeY3, &lowOffsetY, &highOffsetY); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 188 | |
| 189 | if (kernelSizeX < 0 || kernelSizeY < 0) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 190 | return nullptr; |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 191 | } |
| 192 | |
| 193 | if (kernelSizeX == 0 && kernelSizeY == 0) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 194 | offset->fX = inputBounds.x(); |
| 195 | offset->fY = inputBounds.y(); |
| 196 | return input->makeSubset(inputBounds.makeOffset(-inputOffset.x(), |
robertphillips | 2302de9 | 2016-03-24 07:26:32 -0700 | [diff] [blame] | 197 | -inputOffset.y())); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 198 | } |
| 199 | |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 200 | SkBitmap inputBM; |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 201 | |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 202 | if (!input->getROPixels(&inputBM)) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 203 | return nullptr; |
senorblanco | 6404981 | 2016-02-01 10:32:42 -0800 | [diff] [blame] | 204 | } |
| 205 | |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 206 | if (inputBM.colorType() != kN32_SkColorType) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 207 | return nullptr; |
senorblanco | 6404981 | 2016-02-01 10:32:42 -0800 | [diff] [blame] | 208 | } |
senorblanco | 6404981 | 2016-02-01 10:32:42 -0800 | [diff] [blame] | 209 | |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 210 | SkImageInfo info = SkImageInfo::Make(dstBounds.width(), dstBounds.height(), |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 211 | inputBM.colorType(), inputBM.alphaType()); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 212 | |
| 213 | SkBitmap tmp, dst; |
| 214 | if (!tmp.tryAllocPixels(info) || !dst.tryAllocPixels(info)) { |
| 215 | return nullptr; |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 216 | } |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 217 | |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 218 | SkAutoLockPixels inputLock(inputBM), tmpLock(tmp), dstLock(dst); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 219 | |
senorblanco | 29a440d | 2015-11-02 12:55:47 -0800 | [diff] [blame] | 220 | offset->fX = dstBounds.fLeft; |
| 221 | offset->fY = dstBounds.fTop; |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 222 | SkPMColor* t = tmp.getAddr32(0, 0); |
| 223 | SkPMColor* d = dst.getAddr32(0, 0); |
senorblanco@chromium.org | 0cc00c2 | 2013-11-07 18:35:12 +0000 | [diff] [blame] | 224 | int w = dstBounds.width(), h = dstBounds.height(); |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 225 | const SkPMColor* s = inputBM.getAddr32(inputBounds.x() - inputOffset.x(), |
| 226 | inputBounds.y() - inputOffset.y()); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 227 | inputBounds.offset(-dstBounds.x(), -dstBounds.y()); |
senorblanco | 29a440d | 2015-11-02 12:55:47 -0800 | [diff] [blame] | 228 | dstBounds.offset(-dstBounds.x(), -dstBounds.y()); |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 229 | SkIRect inputBoundsT = SkIRect::MakeLTRB(inputBounds.top(), inputBounds.left(), |
| 230 | inputBounds.bottom(), inputBounds.right()); |
senorblanco | 29a440d | 2015-11-02 12:55:47 -0800 | [diff] [blame] | 231 | SkIRect dstBoundsT = SkIRect::MakeWH(dstBounds.height(), dstBounds.width()); |
robertphillips | 6461251 | 2016-04-08 12:10:42 -0700 | [diff] [blame] | 232 | int sw = int(inputBM.rowBytes() >> 2); |
senorblanco@chromium.org | 27eec46 | 2013-11-08 20:49:04 +0000 | [diff] [blame] | 233 | |
mtklein | dce5ce4 | 2015-08-04 08:49:21 -0700 | [diff] [blame] | 234 | /** |
| 235 | * |
| 236 | * In order to make memory accesses cache-friendly, we reorder the passes to |
| 237 | * use contiguous memory reads wherever possible. |
| 238 | * |
| 239 | * For example, the 6 passes of the X-and-Y blur case are rewritten as |
| 240 | * follows. Instead of 3 passes in X and 3 passes in Y, we perform |
| 241 | * 2 passes in X, 1 pass in X transposed to Y on write, 2 passes in X, |
| 242 | * then 1 pass in X transposed to Y on write. |
| 243 | * |
| 244 | * +----+ +----+ +----+ +---+ +---+ +---+ +----+ |
| 245 | * + AB + ----> | AB | ----> | AB | -----> | A | ----> | A | ----> | A | -----> | AB | |
| 246 | * +----+ blurX +----+ blurX +----+ blurXY | B | blurX | B | blurX | B | blurXY +----+ |
| 247 | * +---+ +---+ +---+ |
| 248 | * |
| 249 | * In this way, two of the y-blurs become x-blurs applied to transposed |
| 250 | * images, and all memory reads are contiguous. |
| 251 | */ |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 252 | if (kernelSizeX > 0 && kernelSizeY > 0) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 253 | SkOpts::box_blur_xx(s, sw, inputBounds, t, kernelSizeX, lowOffsetX, highOffsetX, w, h); |
| 254 | SkOpts::box_blur_xx(t, w, dstBounds, d, kernelSizeX, highOffsetX, lowOffsetX, w, h); |
| 255 | SkOpts::box_blur_xy(d, w, dstBounds, t, kernelSizeX3, highOffsetX, highOffsetX, w, h); |
| 256 | SkOpts::box_blur_xx(t, h, dstBoundsT, d, kernelSizeY, lowOffsetY, highOffsetY, h, w); |
| 257 | SkOpts::box_blur_xx(d, h, dstBoundsT, t, kernelSizeY, highOffsetY, lowOffsetY, h, w); |
| 258 | SkOpts::box_blur_xy(t, h, dstBoundsT, d, kernelSizeY3, highOffsetY, highOffsetY, h, w); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 259 | } else if (kernelSizeX > 0) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 260 | SkOpts::box_blur_xx(s, sw, inputBounds, d, kernelSizeX, lowOffsetX, highOffsetX, w, h); |
| 261 | SkOpts::box_blur_xx(d, w, dstBounds, t, kernelSizeX, highOffsetX, lowOffsetX, w, h); |
| 262 | SkOpts::box_blur_xx(t, w, dstBounds, d, kernelSizeX3, highOffsetX, highOffsetX, w, h); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 263 | } else if (kernelSizeY > 0) { |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 264 | SkOpts::box_blur_yx(s, sw, inputBoundsT, d, kernelSizeY, lowOffsetY, highOffsetY, h, w); |
| 265 | SkOpts::box_blur_xx(d, h, dstBoundsT, t, kernelSizeY, highOffsetY, lowOffsetY, h, w); |
| 266 | SkOpts::box_blur_xy(t, h, dstBoundsT, d, kernelSizeY3, highOffsetY, highOffsetY, h, w); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 267 | } |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 268 | |
robertphillips | 3e30227 | 2016-04-20 11:48:36 -0700 | [diff] [blame] | 269 | return SkSpecialImage::MakeFromRaster(SkIRect::MakeWH(dstBounds.width(), |
robertphillips | 1579e3c | 2016-03-24 05:01:23 -0700 | [diff] [blame] | 270 | dstBounds.height()), |
brianosman | 898235c | 2016-04-06 07:38:23 -0700 | [diff] [blame] | 271 | dst, &source->props()); |
senorblanco@chromium.org | ae814c7 | 2011-12-20 20:02:19 +0000 | [diff] [blame] | 272 | } |
| 273 | |
senorblanco@chromium.org | 336d1d7 | 2014-01-27 21:03:17 +0000 | [diff] [blame] | 274 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 275 | SkRect SkBlurImageFilterImpl::computeFastBounds(const SkRect& src) const { |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 276 | SkRect bounds = this->getInput(0) ? this->getInput(0)->computeFastBounds(src) : src; |
| 277 | bounds.outset(SkScalarMul(fSigma.width(), SkIntToScalar(3)), |
| 278 | SkScalarMul(fSigma.height(), SkIntToScalar(3))); |
| 279 | return bounds; |
senorblanco@chromium.org | 336d1d7 | 2014-01-27 21:03:17 +0000 | [diff] [blame] | 280 | } |
senorblanco@chromium.org | c4b12f1 | 2014-02-05 17:51:22 +0000 | [diff] [blame] | 281 | |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 282 | SkIRect SkBlurImageFilterImpl::onFilterNodeBounds(const SkIRect& src, const SkMatrix& ctm, |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 283 | MapDirection) const { |
senorblanco | 6404981 | 2016-02-01 10:32:42 -0800 | [diff] [blame] | 284 | SkVector sigma = map_sigma(fSigma, ctm); |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 285 | return src.makeOutset(SkScalarCeilToInt(SkScalarMul(sigma.x(), SkIntToScalar(3))), |
| 286 | SkScalarCeilToInt(SkScalarMul(sigma.y(), SkIntToScalar(3)))); |
senorblanco@chromium.org | c4b12f1 | 2014-02-05 17:51:22 +0000 | [diff] [blame] | 287 | } |
| 288 | |
robertphillips | f3f5bad | 2014-12-19 13:49:15 -0800 | [diff] [blame] | 289 | #ifndef SK_IGNORE_TO_STRING |
vjiaoblack | e1e5c74 | 2016-08-23 11:13:14 -0700 | [diff] [blame] | 290 | void SkBlurImageFilterImpl::toString(SkString* str) const { |
| 291 | str->appendf("SkBlurImageFilterImpl: ("); |
robertphillips | 33cca88 | 2014-12-22 06:52:04 -0800 | [diff] [blame] | 292 | str->appendf("sigma: (%f, %f) input (", fSigma.fWidth, fSigma.fHeight); |
| 293 | |
| 294 | if (this->getInput(0)) { |
| 295 | this->getInput(0)->toString(str); |
| 296 | } |
| 297 | |
| 298 | str->append("))"); |
robertphillips | f3f5bad | 2014-12-19 13:49:15 -0800 | [diff] [blame] | 299 | } |
| 300 | #endif |