blob: 8d6dbfc2b78d5fd033f59f8af91f379e731f432f [file] [log] [blame]
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +00001/*
2 * Copyright 2012 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
8#include "SkMatrixConvolutionImageFilter.h"
9#include "SkBitmap.h"
10#include "SkColorPriv.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000011#include "SkReadBuffer.h"
robertphillipsdada4dd2016-04-13 04:54:36 -070012#include "SkSpecialImage.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000013#include "SkWriteBuffer.h"
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000014#include "SkRect.h"
senorblanco@chromium.org8640d502012-09-25 14:32:42 +000015#include "SkUnPreMultiply.h"
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000016
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +000017#if SK_SUPPORT_GPU
robertphillipsdada4dd2016-04-13 04:54:36 -070018#include "GrContext.h"
Hal Canary6f6961e2017-01-31 13:50:44 -050019#include "GrTextureProxy.h"
joshualittac977922014-07-22 09:52:11 -070020#include "effects/GrMatrixConvolutionEffect.h"
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +000021#endif
22
senorblanco3a495202014-09-29 07:57:20 -070023// We need to be able to read at most SK_MaxS32 bytes, so divide that
24// by the size of a scalar to know how many scalars we can read.
25static const int32_t gMaxKernelSize = SK_MaxS32 / sizeof(SkScalar);
26
robertphillipsef6a47b2016-04-08 08:01:20 -070027SkMatrixConvolutionImageFilter::SkMatrixConvolutionImageFilter(const SkISize& kernelSize,
28 const SkScalar* kernel,
29 SkScalar gain,
30 SkScalar bias,
31 const SkIPoint& kernelOffset,
32 TileMode tileMode,
33 bool convolveAlpha,
34 sk_sp<SkImageFilter> input,
35 const CropRect* cropRect)
36 : INHERITED(&input, 1, cropRect)
37 , fKernelSize(kernelSize)
38 , fGain(gain)
39 , fBias(bias)
40 , fKernelOffset(kernelOffset)
41 , fTileMode(tileMode)
42 , fConvolveAlpha(convolveAlpha) {
senorblanco3a495202014-09-29 07:57:20 -070043 size_t size = (size_t) sk_64_mul(fKernelSize.width(), fKernelSize.height());
halcanary385fe4d2015-08-26 13:07:48 -070044 fKernel = new SkScalar[size];
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000045 memcpy(fKernel, kernel, size * sizeof(SkScalar));
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +000046 SkASSERT(kernelSize.fWidth >= 1 && kernelSize.fHeight >= 1);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000047 SkASSERT(kernelOffset.fX >= 0 && kernelOffset.fX < kernelSize.fWidth);
48 SkASSERT(kernelOffset.fY >= 0 && kernelOffset.fY < kernelSize.fHeight);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000049}
50
robertphillipsef6a47b2016-04-08 08:01:20 -070051sk_sp<SkImageFilter> SkMatrixConvolutionImageFilter::Make(const SkISize& kernelSize,
52 const SkScalar* kernel,
53 SkScalar gain,
54 SkScalar bias,
55 const SkIPoint& kernelOffset,
56 TileMode tileMode,
57 bool convolveAlpha,
58 sk_sp<SkImageFilter> input,
59 const CropRect* cropRect) {
senorblanco3a495202014-09-29 07:57:20 -070060 if (kernelSize.width() < 1 || kernelSize.height() < 1) {
halcanary96fcdcc2015-08-27 07:41:13 -070061 return nullptr;
senorblanco3a495202014-09-29 07:57:20 -070062 }
63 if (gMaxKernelSize / kernelSize.fWidth < kernelSize.fHeight) {
halcanary96fcdcc2015-08-27 07:41:13 -070064 return nullptr;
senorblanco3a495202014-09-29 07:57:20 -070065 }
66 if (!kernel) {
halcanary96fcdcc2015-08-27 07:41:13 -070067 return nullptr;
senorblanco3a495202014-09-29 07:57:20 -070068 }
sugoi1379b872014-11-17 05:45:55 -080069 if ((kernelOffset.fX < 0) || (kernelOffset.fX >= kernelSize.fWidth) ||
70 (kernelOffset.fY < 0) || (kernelOffset.fY >= kernelSize.fHeight)) {
halcanary96fcdcc2015-08-27 07:41:13 -070071 return nullptr;
sugoi1379b872014-11-17 05:45:55 -080072 }
robertphillipsef6a47b2016-04-08 08:01:20 -070073 return sk_sp<SkImageFilter>(new SkMatrixConvolutionImageFilter(kernelSize, kernel, gain,
74 bias, kernelOffset,
75 tileMode, convolveAlpha,
76 std::move(input), cropRect));
senorblanco3a495202014-09-29 07:57:20 -070077}
78
reed60c9b582016-04-03 09:11:13 -070079sk_sp<SkFlattenable> SkMatrixConvolutionImageFilter::CreateProc(SkReadBuffer& buffer) {
reed9fa60da2014-08-21 07:59:51 -070080 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
81 SkISize kernelSize;
82 kernelSize.fWidth = buffer.readInt();
83 kernelSize.fHeight = buffer.readInt();
84 const int count = buffer.getArrayCount();
85
86 const int64_t kernelArea = sk_64_mul(kernelSize.width(), kernelSize.height());
87 if (!buffer.validate(kernelArea == count)) {
halcanary96fcdcc2015-08-27 07:41:13 -070088 return nullptr;
reed9fa60da2014-08-21 07:59:51 -070089 }
90 SkAutoSTArray<16, SkScalar> kernel(count);
91 if (!buffer.readScalarArray(kernel.get(), count)) {
halcanary96fcdcc2015-08-27 07:41:13 -070092 return nullptr;
reed9fa60da2014-08-21 07:59:51 -070093 }
94 SkScalar gain = buffer.readScalar();
95 SkScalar bias = buffer.readScalar();
96 SkIPoint kernelOffset;
97 kernelOffset.fX = buffer.readInt();
98 kernelOffset.fY = buffer.readInt();
99 TileMode tileMode = (TileMode)buffer.readInt();
100 bool convolveAlpha = buffer.readBool();
robertphillipsef6a47b2016-04-08 08:01:20 -0700101 return Make(kernelSize, kernel.get(), gain, bias, kernelOffset, tileMode,
102 convolveAlpha, common.getInput(0), &common.cropRect());
reed9fa60da2014-08-21 07:59:51 -0700103}
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000104
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +0000105void SkMatrixConvolutionImageFilter::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000106 this->INHERITED::flatten(buffer);
107 buffer.writeInt(fKernelSize.fWidth);
108 buffer.writeInt(fKernelSize.fHeight);
109 buffer.writeScalarArray(fKernel, fKernelSize.fWidth * fKernelSize.fHeight);
110 buffer.writeScalar(fGain);
111 buffer.writeScalar(fBias);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000112 buffer.writeInt(fKernelOffset.fX);
113 buffer.writeInt(fKernelOffset.fY);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000114 buffer.writeInt((int) fTileMode);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000115 buffer.writeBool(fConvolveAlpha);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000116}
117
118SkMatrixConvolutionImageFilter::~SkMatrixConvolutionImageFilter() {
119 delete[] fKernel;
120}
121
122class UncheckedPixelFetcher {
123public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000124 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000125 return *src.getAddr32(x, y);
126 }
127};
128
129class ClampPixelFetcher {
130public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000131 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
bungeman62ce0302015-08-28 09:09:32 -0700132 x = SkTPin(x, bounds.fLeft, bounds.fRight - 1);
133 y = SkTPin(y, bounds.fTop, bounds.fBottom - 1);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000134 return *src.getAddr32(x, y);
135 }
136};
137
138class RepeatPixelFetcher {
139public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000140 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
141 x = (x - bounds.left()) % bounds.width() + bounds.left();
142 y = (y - bounds.top()) % bounds.height() + bounds.top();
143 if (x < bounds.left()) {
144 x += bounds.width();
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000145 }
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000146 if (y < bounds.top()) {
147 y += bounds.height();
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000148 }
149 return *src.getAddr32(x, y);
150 }
151};
152
153class ClampToBlackPixelFetcher {
154public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000155 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
156 if (x < bounds.fLeft || x >= bounds.fRight || y < bounds.fTop || y >= bounds.fBottom) {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000157 return 0;
158 } else {
159 return *src.getAddr32(x, y);
160 }
161 }
162};
163
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000164template<class PixelFetcher, bool convolveAlpha>
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000165void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src,
166 SkBitmap* result,
senorblanco@chromium.org8c7372b2014-05-02 19:13:11 +0000167 const SkIRect& r,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000168 const SkIRect& bounds) const {
senorblanco@chromium.org8c7372b2014-05-02 19:13:11 +0000169 SkIRect rect(r);
170 if (!rect.intersect(bounds)) {
senorblanco@chromium.org91957432014-05-01 14:03:41 +0000171 return;
172 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000173 for (int y = rect.fTop; y < rect.fBottom; ++y) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000174 SkPMColor* dptr = result->getAddr32(rect.fLeft - bounds.fLeft, y - bounds.fTop);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000175 for (int x = rect.fLeft; x < rect.fRight; ++x) {
176 SkScalar sumA = 0, sumR = 0, sumG = 0, sumB = 0;
177 for (int cy = 0; cy < fKernelSize.fHeight; cy++) {
178 for (int cx = 0; cx < fKernelSize.fWidth; cx++) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000179 SkPMColor s = PixelFetcher::fetch(src,
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000180 x + cx - fKernelOffset.fX,
181 y + cy - fKernelOffset.fY,
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000182 bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000183 SkScalar k = fKernel[cy * fKernelSize.fWidth + cx];
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000184 if (convolveAlpha) {
185 sumA += SkScalarMul(SkIntToScalar(SkGetPackedA32(s)), k);
186 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000187 sumR += SkScalarMul(SkIntToScalar(SkGetPackedR32(s)), k);
188 sumG += SkScalarMul(SkIntToScalar(SkGetPackedG32(s)), k);
189 sumB += SkScalarMul(SkIntToScalar(SkGetPackedB32(s)), k);
190 }
191 }
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000192 int a = convolveAlpha
193 ? SkClampMax(SkScalarFloorToInt(SkScalarMul(sumA, fGain) + fBias), 255)
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000194 : 255;
senorblanco@chromium.orgcc9471c2012-09-20 17:59:49 +0000195 int r = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumR, fGain) + fBias), a);
196 int g = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumG, fGain) + fBias), a);
197 int b = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumB, fGain) + fBias), a);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000198 if (!convolveAlpha) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000199 a = SkGetPackedA32(PixelFetcher::fetch(src, x, y, bounds));
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000200 *dptr++ = SkPreMultiplyARGB(a, r, g, b);
201 } else {
202 *dptr++ = SkPackARGB32(a, r, g, b);
203 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000204 }
205 }
206}
207
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000208template<class PixelFetcher>
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000209void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src,
210 SkBitmap* result,
211 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000212 const SkIRect& bounds) const {
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000213 if (fConvolveAlpha) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000214 filterPixels<PixelFetcher, true>(src, result, rect, bounds);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000215 } else {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000216 filterPixels<PixelFetcher, false>(src, result, rect, bounds);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000217 }
218}
219
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000220void SkMatrixConvolutionImageFilter::filterInteriorPixels(const SkBitmap& src,
221 SkBitmap* result,
222 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000223 const SkIRect& bounds) const {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000224 filterPixels<UncheckedPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000225}
226
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000227void SkMatrixConvolutionImageFilter::filterBorderPixels(const SkBitmap& src,
228 SkBitmap* result,
229 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000230 const SkIRect& bounds) const {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000231 switch (fTileMode) {
232 case kClamp_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000233 filterPixels<ClampPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000234 break;
235 case kRepeat_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000236 filterPixels<RepeatPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000237 break;
238 case kClampToBlack_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000239 filterPixels<ClampToBlackPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000240 break;
241 }
242}
243
senorblanco@chromium.org377c14a2013-02-04 22:57:21 +0000244// FIXME: This should be refactored to SkImageFilterUtils for
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000245// use by other filters. For now, we assume the input is always
246// premultiplied and unpremultiply it
robertphillipsdada4dd2016-04-13 04:54:36 -0700247static SkBitmap unpremultiply_bitmap(const SkBitmap& src)
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000248{
249 SkAutoLockPixels alp(src);
250 if (!src.getPixels()) {
251 return SkBitmap();
252 }
robertphillipsdada4dd2016-04-13 04:54:36 -0700253
254 const SkImageInfo info = SkImageInfo::MakeN32(src.width(), src.height(), src.alphaType());
255 SkBitmap result;
256 if (!result.tryAllocPixels(info)) {
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000257 return SkBitmap();
258 }
robertphillipsdada4dd2016-04-13 04:54:36 -0700259 SkAutoLockPixels resultLock(result);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000260 for (int y = 0; y < src.height(); ++y) {
261 const uint32_t* srcRow = src.getAddr32(0, y);
262 uint32_t* dstRow = result.getAddr32(0, y);
263 for (int x = 0; x < src.width(); ++x) {
264 dstRow[x] = SkUnPreMultiply::PMColorToColor(srcRow[x]);
265 }
266 }
267 return result;
268}
269
robertphillipsdada4dd2016-04-13 04:54:36 -0700270#if SK_SUPPORT_GPU
senorblanco@chromium.org68400762013-05-24 15:04:07 +0000271
robertphillipsdada4dd2016-04-13 04:54:36 -0700272static GrTextureDomain::Mode convert_tilemodes(SkMatrixConvolutionImageFilter::TileMode tileMode) {
273 switch (tileMode) {
274 case SkMatrixConvolutionImageFilter::kClamp_TileMode:
275 return GrTextureDomain::kClamp_Mode;
276 case SkMatrixConvolutionImageFilter::kRepeat_TileMode:
277 return GrTextureDomain::kRepeat_Mode;
278 case SkMatrixConvolutionImageFilter::kClampToBlack_TileMode:
279 return GrTextureDomain::kDecal_Mode;
280 default:
281 SkASSERT(false);
282 }
283 return GrTextureDomain::kIgnore_Mode;
284}
robertphillipsdada4dd2016-04-13 04:54:36 -0700285#endif
286
287sk_sp<SkSpecialImage> SkMatrixConvolutionImageFilter::onFilterImage(SkSpecialImage* source,
288 const Context& ctx,
289 SkIPoint* offset) const {
290 SkIPoint inputOffset = SkIPoint::Make(0, 0);
291 sk_sp<SkSpecialImage> input(this->filterInput(0, source, ctx, &inputOffset));
292 if (!input) {
293 return nullptr;
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000294 }
295
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000296 SkIRect bounds;
robertphillipsdada4dd2016-04-13 04:54:36 -0700297 input = this->applyCropRect(this->mapContext(ctx), input.get(), &inputOffset, &bounds);
298 if (!input) {
299 return nullptr;
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000300 }
301
robertphillipsdada4dd2016-04-13 04:54:36 -0700302#if SK_SUPPORT_GPU
303 // Note: if the kernel is too big, the GPU path falls back to SW
304 if (source->isTextureBacked() &&
305 fKernelSize.width() * fKernelSize.height() <= MAX_KERNEL_SIZE) {
306 GrContext* context = source->getContext();
307
Brian Osman615d66d2016-12-29 09:18:20 -0500308 // Ensure the input is in the destination color space. Typically applyCropRect will have
309 // called pad_image to account for our dilation of bounds, so the result will already be
310 // moved to the destination color space. If a filter DAG avoids that, then we use this
311 // fall-back, which saves us from having to do the xform during the filter itself.
312 input = ImageToColorSpace(input.get(), ctx.outputProperties());
Brian Osman29437eb2016-12-27 12:50:37 -0500313
robertphillipsdada4dd2016-04-13 04:54:36 -0700314 sk_sp<GrTexture> inputTexture(input->asTextureRef(context));
315 SkASSERT(inputTexture);
316
317 offset->fX = bounds.left();
318 offset->fY = bounds.top();
319 bounds.offset(-inputOffset);
320
bungeman06ca8ec2016-06-09 08:01:03 -0700321 sk_sp<GrFragmentProcessor> fp(GrMatrixConvolutionEffect::Make(inputTexture.get(),
robertphillipsdada4dd2016-04-13 04:54:36 -0700322 bounds,
323 fKernelSize,
324 fKernel,
325 fGain,
326 fBias,
327 fKernelOffset,
328 convert_tilemodes(fTileMode),
329 fConvolveAlpha));
330 if (!fp) {
331 return nullptr;
332 }
333
brianosman2a75e5d2016-09-22 07:15:37 -0700334 return DrawWithFP(context, std::move(fp), bounds, ctx.outputProperties());
robertphillipsdada4dd2016-04-13 04:54:36 -0700335 }
336#endif
337
338 SkBitmap inputBM;
339
340 if (!input->getROPixels(&inputBM)) {
341 return nullptr;
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000342 }
343
robertphillipsdada4dd2016-04-13 04:54:36 -0700344 if (inputBM.colorType() != kN32_SkColorType) {
345 return nullptr;
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000346 }
347
robertphillipsdada4dd2016-04-13 04:54:36 -0700348 if (!fConvolveAlpha && !inputBM.isOpaque()) {
349 inputBM = unpremultiply_bitmap(inputBM);
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +0000350 }
robertphillipsdada4dd2016-04-13 04:54:36 -0700351
352 SkAutoLockPixels alp(inputBM);
353 if (!inputBM.getPixels()) {
354 return nullptr;
355 }
356
357 const SkImageInfo info = SkImageInfo::MakeN32(bounds.width(), bounds.height(),
358 inputBM.alphaType());
359
360 SkBitmap dst;
361 if (!dst.tryAllocPixels(info)) {
362 return nullptr;
363 }
364
365 SkAutoLockPixels dstLock(dst);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000366
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000367 offset->fX = bounds.fLeft;
368 offset->fY = bounds.fTop;
robertphillipsdada4dd2016-04-13 04:54:36 -0700369 bounds.offset(-inputOffset);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000370 SkIRect interior = SkIRect::MakeXYWH(bounds.left() + fKernelOffset.fX,
371 bounds.top() + fKernelOffset.fY,
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000372 bounds.width() - fKernelSize.fWidth + 1,
373 bounds.height() - fKernelSize.fHeight + 1);
374 SkIRect top = SkIRect::MakeLTRB(bounds.left(), bounds.top(), bounds.right(), interior.top());
375 SkIRect bottom = SkIRect::MakeLTRB(bounds.left(), interior.bottom(),
376 bounds.right(), bounds.bottom());
377 SkIRect left = SkIRect::MakeLTRB(bounds.left(), interior.top(),
378 interior.left(), interior.bottom());
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000379 SkIRect right = SkIRect::MakeLTRB(interior.right(), interior.top(),
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000380 bounds.right(), interior.bottom());
robertphillipsdada4dd2016-04-13 04:54:36 -0700381 this->filterBorderPixels(inputBM, &dst, top, bounds);
382 this->filterBorderPixels(inputBM, &dst, left, bounds);
383 this->filterInteriorPixels(inputBM, &dst, interior, bounds);
384 this->filterBorderPixels(inputBM, &dst, right, bounds);
385 this->filterBorderPixels(inputBM, &dst, bottom, bounds);
robertphillips3e302272016-04-20 11:48:36 -0700386 return SkSpecialImage::MakeFromRaster(SkIRect::MakeWH(bounds.width(), bounds.height()),
robertphillipsdada4dd2016-04-13 04:54:36 -0700387 dst);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000388}
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000389
senorblancoe5e79842016-03-21 14:51:59 -0700390SkIRect SkMatrixConvolutionImageFilter::onFilterNodeBounds(const SkIRect& src, const SkMatrix& ctm,
391 MapDirection direction) const {
392 SkIRect dst = src;
senorblancodb64af32015-12-09 10:11:43 -0800393 int w = fKernelSize.width() - 1, h = fKernelSize.height() - 1;
senorblancoe5e79842016-03-21 14:51:59 -0700394 dst.fRight += w;
395 dst.fBottom += h;
senorblancodb64af32015-12-09 10:11:43 -0800396 if (kReverse_MapDirection == direction) {
senorblancoe5e79842016-03-21 14:51:59 -0700397 dst.offset(-fKernelOffset);
senorblancodb64af32015-12-09 10:11:43 -0800398 } else {
senorblancoe5e79842016-03-21 14:51:59 -0700399 dst.offset(fKernelOffset - SkIPoint::Make(w, h));
senorblanco@chromium.org0a5c2332014-04-29 15:20:39 +0000400 }
senorblancoe5e79842016-03-21 14:51:59 -0700401 return dst;
senorblanco@chromium.org0a5c2332014-04-29 15:20:39 +0000402}
403
senorblanco6db0a7b2016-04-01 16:41:10 -0700404bool SkMatrixConvolutionImageFilter::affectsTransparentBlack() const {
senorblancoa544eda2015-12-07 07:48:34 -0800405 // Because the kernel is applied in device-space, we have no idea what
406 // pixels it will affect in object-space.
senorblanco6db0a7b2016-04-01 16:41:10 -0700407 return true;
senorblancoa544eda2015-12-07 07:48:34 -0800408}
409
robertphillipsf3f5bad2014-12-19 13:49:15 -0800410#ifndef SK_IGNORE_TO_STRING
411void SkMatrixConvolutionImageFilter::toString(SkString* str) const {
412 str->appendf("SkMatrixConvolutionImageFilter: (");
413 str->appendf("size: (%d,%d) kernel: (", fKernelSize.width(), fKernelSize.height());
414 for (int y = 0; y < fKernelSize.height(); y++) {
415 for (int x = 0; x < fKernelSize.width(); x++) {
416 str->appendf("%f ", fKernel[y * fKernelSize.width() + x]);
417 }
418 }
419 str->appendf(")");
420 str->appendf("gain: %f bias: %f ", fGain, fBias);
421 str->appendf("offset: (%d, %d) ", fKernelOffset.fX, fKernelOffset.fY);
422 str->appendf("convolveAlpha: %s", fConvolveAlpha ? "true" : "false");
423 str->append(")");
424}
425#endif