blob: 0d14caf1314b3e4011e149802d165e259b7abfaf [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"
senorblanco1d3ff432015-10-20 10:17:34 -070011#include "SkDevice.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000012#include "SkReadBuffer.h"
13#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
joshualittac977922014-07-22 09:52:11 -070018#include "effects/GrMatrixConvolutionEffect.h"
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +000019#endif
20
senorblanco3a495202014-09-29 07:57:20 -070021// We need to be able to read at most SK_MaxS32 bytes, so divide that
22// by the size of a scalar to know how many scalars we can read.
23static const int32_t gMaxKernelSize = SK_MaxS32 / sizeof(SkScalar);
24
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +000025SkMatrixConvolutionImageFilter::SkMatrixConvolutionImageFilter(
26 const SkISize& kernelSize,
27 const SkScalar* kernel,
28 SkScalar gain,
29 SkScalar bias,
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000030 const SkIPoint& kernelOffset,
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +000031 TileMode tileMode,
32 bool convolveAlpha,
33 SkImageFilter* input,
senorblanco24e06d52015-03-18 12:11:33 -070034 const CropRect* cropRect)
35 : INHERITED(1, &input, cropRect),
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000036 fKernelSize(kernelSize),
37 fGain(gain),
38 fBias(bias),
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000039 fKernelOffset(kernelOffset),
senorblanco@chromium.org8640d502012-09-25 14:32:42 +000040 fTileMode(tileMode),
41 fConvolveAlpha(convolveAlpha) {
senorblanco3a495202014-09-29 07:57:20 -070042 size_t size = (size_t) sk_64_mul(fKernelSize.width(), fKernelSize.height());
halcanary385fe4d2015-08-26 13:07:48 -070043 fKernel = new SkScalar[size];
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000044 memcpy(fKernel, kernel, size * sizeof(SkScalar));
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +000045 SkASSERT(kernelSize.fWidth >= 1 && kernelSize.fHeight >= 1);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +000046 SkASSERT(kernelOffset.fX >= 0 && kernelOffset.fX < kernelSize.fWidth);
47 SkASSERT(kernelOffset.fY >= 0 && kernelOffset.fY < kernelSize.fHeight);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +000048}
49
reed5e1ddb12015-12-21 08:52:45 -080050SkImageFilter* SkMatrixConvolutionImageFilter::Create(
senorblanco3a495202014-09-29 07:57:20 -070051 const SkISize& kernelSize,
52 const SkScalar* kernel,
53 SkScalar gain,
54 SkScalar bias,
55 const SkIPoint& kernelOffset,
56 TileMode tileMode,
57 bool convolveAlpha,
58 SkImageFilter* input,
senorblanco24e06d52015-03-18 12:11:33 -070059 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 }
halcanary385fe4d2015-08-26 13:07:48 -070073 return new SkMatrixConvolutionImageFilter(kernelSize, kernel, gain, bias, kernelOffset,
74 tileMode, convolveAlpha, input, cropRect);
senorblanco3a495202014-09-29 07:57:20 -070075}
76
reed9fa60da2014-08-21 07:59:51 -070077SkFlattenable* SkMatrixConvolutionImageFilter::CreateProc(SkReadBuffer& buffer) {
78 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
79 SkISize kernelSize;
80 kernelSize.fWidth = buffer.readInt();
81 kernelSize.fHeight = buffer.readInt();
82 const int count = buffer.getArrayCount();
83
84 const int64_t kernelArea = sk_64_mul(kernelSize.width(), kernelSize.height());
85 if (!buffer.validate(kernelArea == count)) {
halcanary96fcdcc2015-08-27 07:41:13 -070086 return nullptr;
reed9fa60da2014-08-21 07:59:51 -070087 }
88 SkAutoSTArray<16, SkScalar> kernel(count);
89 if (!buffer.readScalarArray(kernel.get(), count)) {
halcanary96fcdcc2015-08-27 07:41:13 -070090 return nullptr;
reed9fa60da2014-08-21 07:59:51 -070091 }
92 SkScalar gain = buffer.readScalar();
93 SkScalar bias = buffer.readScalar();
94 SkIPoint kernelOffset;
95 kernelOffset.fX = buffer.readInt();
96 kernelOffset.fY = buffer.readInt();
97 TileMode tileMode = (TileMode)buffer.readInt();
98 bool convolveAlpha = buffer.readBool();
99 return Create(kernelSize, kernel.get(), gain, bias, kernelOffset, tileMode, convolveAlpha,
senorblanco24e06d52015-03-18 12:11:33 -0700100 common.getInput(0), &common.cropRect());
reed9fa60da2014-08-21 07:59:51 -0700101}
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000102
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +0000103void SkMatrixConvolutionImageFilter::flatten(SkWriteBuffer& buffer) const {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000104 this->INHERITED::flatten(buffer);
105 buffer.writeInt(fKernelSize.fWidth);
106 buffer.writeInt(fKernelSize.fHeight);
107 buffer.writeScalarArray(fKernel, fKernelSize.fWidth * fKernelSize.fHeight);
108 buffer.writeScalar(fGain);
109 buffer.writeScalar(fBias);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000110 buffer.writeInt(fKernelOffset.fX);
111 buffer.writeInt(fKernelOffset.fY);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000112 buffer.writeInt((int) fTileMode);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000113 buffer.writeBool(fConvolveAlpha);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000114}
115
116SkMatrixConvolutionImageFilter::~SkMatrixConvolutionImageFilter() {
117 delete[] fKernel;
118}
119
120class UncheckedPixelFetcher {
121public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000122 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000123 return *src.getAddr32(x, y);
124 }
125};
126
127class ClampPixelFetcher {
128public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000129 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
bungeman62ce0302015-08-28 09:09:32 -0700130 x = SkTPin(x, bounds.fLeft, bounds.fRight - 1);
131 y = SkTPin(y, bounds.fTop, bounds.fBottom - 1);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000132 return *src.getAddr32(x, y);
133 }
134};
135
136class RepeatPixelFetcher {
137public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000138 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
139 x = (x - bounds.left()) % bounds.width() + bounds.left();
140 y = (y - bounds.top()) % bounds.height() + bounds.top();
141 if (x < bounds.left()) {
142 x += bounds.width();
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000143 }
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000144 if (y < bounds.top()) {
145 y += bounds.height();
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000146 }
147 return *src.getAddr32(x, y);
148 }
149};
150
151class ClampToBlackPixelFetcher {
152public:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000153 static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) {
154 if (x < bounds.fLeft || x >= bounds.fRight || y < bounds.fTop || y >= bounds.fBottom) {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000155 return 0;
156 } else {
157 return *src.getAddr32(x, y);
158 }
159 }
160};
161
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000162template<class PixelFetcher, bool convolveAlpha>
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000163void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src,
164 SkBitmap* result,
senorblanco@chromium.org8c7372b2014-05-02 19:13:11 +0000165 const SkIRect& r,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000166 const SkIRect& bounds) const {
senorblanco@chromium.org8c7372b2014-05-02 19:13:11 +0000167 SkIRect rect(r);
168 if (!rect.intersect(bounds)) {
senorblanco@chromium.org91957432014-05-01 14:03:41 +0000169 return;
170 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000171 for (int y = rect.fTop; y < rect.fBottom; ++y) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000172 SkPMColor* dptr = result->getAddr32(rect.fLeft - bounds.fLeft, y - bounds.fTop);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000173 for (int x = rect.fLeft; x < rect.fRight; ++x) {
174 SkScalar sumA = 0, sumR = 0, sumG = 0, sumB = 0;
175 for (int cy = 0; cy < fKernelSize.fHeight; cy++) {
176 for (int cx = 0; cx < fKernelSize.fWidth; cx++) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000177 SkPMColor s = PixelFetcher::fetch(src,
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000178 x + cx - fKernelOffset.fX,
179 y + cy - fKernelOffset.fY,
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000180 bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000181 SkScalar k = fKernel[cy * fKernelSize.fWidth + cx];
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000182 if (convolveAlpha) {
183 sumA += SkScalarMul(SkIntToScalar(SkGetPackedA32(s)), k);
184 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000185 sumR += SkScalarMul(SkIntToScalar(SkGetPackedR32(s)), k);
186 sumG += SkScalarMul(SkIntToScalar(SkGetPackedG32(s)), k);
187 sumB += SkScalarMul(SkIntToScalar(SkGetPackedB32(s)), k);
188 }
189 }
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000190 int a = convolveAlpha
191 ? SkClampMax(SkScalarFloorToInt(SkScalarMul(sumA, fGain) + fBias), 255)
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000192 : 255;
senorblanco@chromium.orgcc9471c2012-09-20 17:59:49 +0000193 int r = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumR, fGain) + fBias), a);
194 int g = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumG, fGain) + fBias), a);
195 int b = SkClampMax(SkScalarFloorToInt(SkScalarMul(sumB, fGain) + fBias), a);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000196 if (!convolveAlpha) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000197 a = SkGetPackedA32(PixelFetcher::fetch(src, x, y, bounds));
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000198 *dptr++ = SkPreMultiplyARGB(a, r, g, b);
199 } else {
200 *dptr++ = SkPackARGB32(a, r, g, b);
201 }
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000202 }
203 }
204}
205
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000206template<class PixelFetcher>
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000207void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src,
208 SkBitmap* result,
209 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000210 const SkIRect& bounds) const {
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000211 if (fConvolveAlpha) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000212 filterPixels<PixelFetcher, true>(src, result, rect, bounds);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000213 } else {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000214 filterPixels<PixelFetcher, false>(src, result, rect, bounds);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000215 }
216}
217
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000218void SkMatrixConvolutionImageFilter::filterInteriorPixels(const SkBitmap& src,
219 SkBitmap* result,
220 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000221 const SkIRect& bounds) const {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000222 filterPixels<UncheckedPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000223}
224
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000225void SkMatrixConvolutionImageFilter::filterBorderPixels(const SkBitmap& src,
226 SkBitmap* result,
227 const SkIRect& rect,
commit-bot@chromium.orgae761f72014-02-05 22:32:02 +0000228 const SkIRect& bounds) const {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000229 switch (fTileMode) {
230 case kClamp_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000231 filterPixels<ClampPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000232 break;
233 case kRepeat_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000234 filterPixels<RepeatPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000235 break;
236 case kClampToBlack_TileMode:
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000237 filterPixels<ClampToBlackPixelFetcher>(src, result, rect, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000238 break;
239 }
240}
241
senorblanco@chromium.org377c14a2013-02-04 22:57:21 +0000242// FIXME: This should be refactored to SkImageFilterUtils for
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000243// use by other filters. For now, we assume the input is always
244// premultiplied and unpremultiply it
senorblanco1d3ff432015-10-20 10:17:34 -0700245static SkBitmap unpremultiplyBitmap(SkImageFilter::Proxy* proxy, const SkBitmap& src)
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000246{
247 SkAutoLockPixels alp(src);
248 if (!src.getPixels()) {
249 return SkBitmap();
250 }
senorblanco1d3ff432015-10-20 10:17:34 -0700251 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(src.width(), src.height()));
252 if (!device) {
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000253 return SkBitmap();
254 }
senorblanco1d3ff432015-10-20 10:17:34 -0700255 SkBitmap result = device->accessBitmap(false);
256 SkAutoLockPixels alp_result(result);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000257 for (int y = 0; y < src.height(); ++y) {
258 const uint32_t* srcRow = src.getAddr32(0, y);
259 uint32_t* dstRow = result.getAddr32(0, y);
260 for (int x = 0; x < src.width(); ++x) {
261 dstRow[x] = SkUnPreMultiply::PMColorToColor(srcRow[x]);
262 }
263 }
264 return result;
265}
266
robertphillips48e78462016-02-17 13:57:16 -0800267bool SkMatrixConvolutionImageFilter::onFilterImageDeprecated(Proxy* proxy,
268 const SkBitmap& source,
269 const Context& ctx,
270 SkBitmap* result,
271 SkIPoint* offset) const {
senorblanco@chromium.org68400762013-05-24 15:04:07 +0000272 SkBitmap src = source;
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000273 SkIPoint srcOffset = SkIPoint::Make(0, 0);
robertphillips48e78462016-02-17 13:57:16 -0800274 if (!this->filterInputDeprecated(0, proxy, source, ctx, &src, &srcOffset)) {
senorblanco@chromium.org68400762013-05-24 15:04:07 +0000275 return false;
276 }
277
commit-bot@chromium.org28fcae22014-04-11 17:15:40 +0000278 if (src.colorType() != kN32_SkColorType) {
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000279 return false;
280 }
281
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000282 SkIRect bounds;
robertphillips48e78462016-02-17 13:57:16 -0800283 if (!this->applyCropRectDeprecated(this->mapContext(ctx), proxy, src, &srcOffset,
284 &bounds, &src)) {
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000285 return false;
286 }
287
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000288 if (!fConvolveAlpha && !src.isOpaque()) {
senorblanco1d3ff432015-10-20 10:17:34 -0700289 src = unpremultiplyBitmap(proxy, src);
senorblanco@chromium.org8640d502012-09-25 14:32:42 +0000290 }
291
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000292 SkAutoLockPixels alp(src);
293 if (!src.getPixels()) {
294 return false;
295 }
296
senorblanco1d3ff432015-10-20 10:17:34 -0700297 SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(bounds.width(), bounds.height()));
298 if (!device) {
commit-bot@chromium.orgcd3b15c2013-12-04 17:06:49 +0000299 return false;
300 }
senorblanco1d3ff432015-10-20 10:17:34 -0700301 *result = device->accessBitmap(false);
302 SkAutoLockPixels alp_result(*result);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000303
senorblanco@chromium.org6776b822014-01-03 21:48:22 +0000304 offset->fX = bounds.fLeft;
305 offset->fY = bounds.fTop;
306 bounds.offset(-srcOffset);
commit-bot@chromium.orgcac5fd52014-03-10 10:51:58 +0000307 SkIRect interior = SkIRect::MakeXYWH(bounds.left() + fKernelOffset.fX,
308 bounds.top() + fKernelOffset.fY,
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000309 bounds.width() - fKernelSize.fWidth + 1,
310 bounds.height() - fKernelSize.fHeight + 1);
311 SkIRect top = SkIRect::MakeLTRB(bounds.left(), bounds.top(), bounds.right(), interior.top());
312 SkIRect bottom = SkIRect::MakeLTRB(bounds.left(), interior.bottom(),
313 bounds.right(), bounds.bottom());
314 SkIRect left = SkIRect::MakeLTRB(bounds.left(), interior.top(),
315 interior.left(), interior.bottom());
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000316 SkIRect right = SkIRect::MakeLTRB(interior.right(), interior.top(),
senorblanco@chromium.org7938bae2013-10-18 20:08:14 +0000317 bounds.right(), interior.bottom());
318 filterBorderPixels(src, result, top, bounds);
319 filterBorderPixels(src, result, left, bounds);
320 filterInteriorPixels(src, result, interior, bounds);
321 filterBorderPixels(src, result, right, bounds);
322 filterBorderPixels(src, result, bottom, bounds);
senorblanco@chromium.org5faa2dc2012-09-18 20:32:34 +0000323 return true;
324}
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000325
senorblancodb64af32015-12-09 10:11:43 -0800326void SkMatrixConvolutionImageFilter::onFilterNodeBounds(const SkIRect& src, const SkMatrix& ctm,
327 SkIRect* dst, MapDirection direction) const {
328 *dst = src;
329 int w = fKernelSize.width() - 1, h = fKernelSize.height() - 1;
330 dst->fRight += w;
331 dst->fBottom += h;
332 if (kReverse_MapDirection == direction) {
333 dst->offset(-fKernelOffset);
334 } else {
335 dst->offset(fKernelOffset - SkIPoint::Make(w, h));
senorblanco@chromium.org0a5c2332014-04-29 15:20:39 +0000336 }
senorblanco@chromium.org0a5c2332014-04-29 15:20:39 +0000337}
338
senorblancoa544eda2015-12-07 07:48:34 -0800339bool SkMatrixConvolutionImageFilter::canComputeFastBounds() const {
340 // Because the kernel is applied in device-space, we have no idea what
341 // pixels it will affect in object-space.
342 return false;
343}
344
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000345#if SK_SUPPORT_GPU
346
joshualitt5ae5fc52014-07-29 12:59:27 -0700347static GrTextureDomain::Mode convert_tilemodes(
joshualittac977922014-07-22 09:52:11 -0700348 SkMatrixConvolutionImageFilter::TileMode tileMode) {
joshualitt5ae5fc52014-07-29 12:59:27 -0700349 switch (tileMode) {
350 case SkMatrixConvolutionImageFilter::kClamp_TileMode:
351 return GrTextureDomain::kClamp_Mode;
352 case SkMatrixConvolutionImageFilter::kRepeat_TileMode:
353 return GrTextureDomain::kRepeat_Mode;
354 case SkMatrixConvolutionImageFilter::kClampToBlack_TileMode:
355 return GrTextureDomain::kDecal_Mode;
356 default:
357 SkASSERT(false);
358 }
359 return GrTextureDomain::kIgnore_Mode;
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000360}
361
joshualittb0a8a372014-09-23 09:50:21 -0700362bool SkMatrixConvolutionImageFilter::asFragmentProcessor(GrFragmentProcessor** fp,
363 GrTexture* texture,
364 const SkMatrix&,
365 const SkIRect& bounds) const {
366 if (!fp) {
senorblanco@chromium.orgd043cce2013-04-08 19:43:22 +0000367 return fKernelSize.width() * fKernelSize.height() <= MAX_KERNEL_SIZE;
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000368 }
senorblanco@chromium.orgd043cce2013-04-08 19:43:22 +0000369 SkASSERT(fKernelSize.width() * fKernelSize.height() <= MAX_KERNEL_SIZE);
bsalomon4a339522015-10-06 08:40:50 -0700370 *fp = GrMatrixConvolutionEffect::Create(texture,
joshualittb0a8a372014-09-23 09:50:21 -0700371 bounds,
372 fKernelSize,
373 fKernel,
374 fGain,
375 fBias,
376 fKernelOffset,
377 convert_tilemodes(fTileMode),
378 fConvolveAlpha);
senorblanco@chromium.orgd043cce2013-04-08 19:43:22 +0000379 return true;
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000380}
senorblanco@chromium.org3bc16c82012-10-04 17:18:20 +0000381#endif
robertphillipsf3f5bad2014-12-19 13:49:15 -0800382
383#ifndef SK_IGNORE_TO_STRING
384void SkMatrixConvolutionImageFilter::toString(SkString* str) const {
385 str->appendf("SkMatrixConvolutionImageFilter: (");
386 str->appendf("size: (%d,%d) kernel: (", fKernelSize.width(), fKernelSize.height());
387 for (int y = 0; y < fKernelSize.height(); y++) {
388 for (int x = 0; x < fKernelSize.width(); x++) {
389 str->appendf("%f ", fKernel[y * fKernelSize.width() + x]);
390 }
391 }
392 str->appendf(")");
393 str->appendf("gain: %f bias: %f ", fGain, fBias);
394 str->appendf("offset: (%d, %d) ", fKernelOffset.fX, fKernelOffset.fY);
395 str->appendf("convolveAlpha: %s", fConvolveAlpha ? "true" : "false");
396 str->append(")");
397}
398#endif