senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 1 | /* |
| 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.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 11 | #include "SkReadBuffer.h" |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 12 | #include "SkSpecialImage.h" |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 13 | #include "SkWriteBuffer.h" |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 14 | #include "SkRect.h" |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 15 | #include "SkUnPreMultiply.h" |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 16 | |
senorblanco@chromium.org | 3bc16c8 | 2012-10-04 17:18:20 +0000 | [diff] [blame] | 17 | #if SK_SUPPORT_GPU |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 18 | #include "GrContext.h" |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 19 | #include "GrTextureProxy.h" |
joshualitt | ac97792 | 2014-07-22 09:52:11 -0700 | [diff] [blame] | 20 | #include "effects/GrMatrixConvolutionEffect.h" |
senorblanco@chromium.org | 3bc16c8 | 2012-10-04 17:18:20 +0000 | [diff] [blame] | 21 | #endif |
| 22 | |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 23 | // 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. |
| 25 | static const int32_t gMaxKernelSize = SK_MaxS32 / sizeof(SkScalar); |
| 26 | |
robertphillips | ef6a47b | 2016-04-08 08:01:20 -0700 | [diff] [blame] | 27 | SkMatrixConvolutionImageFilter::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) { |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 43 | size_t size = (size_t) sk_64_mul(fKernelSize.width(), fKernelSize.height()); |
halcanary | 385fe4d | 2015-08-26 13:07:48 -0700 | [diff] [blame] | 44 | fKernel = new SkScalar[size]; |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 45 | memcpy(fKernel, kernel, size * sizeof(SkScalar)); |
senorblanco@chromium.org | 3bc16c8 | 2012-10-04 17:18:20 +0000 | [diff] [blame] | 46 | SkASSERT(kernelSize.fWidth >= 1 && kernelSize.fHeight >= 1); |
commit-bot@chromium.org | cac5fd5 | 2014-03-10 10:51:58 +0000 | [diff] [blame] | 47 | SkASSERT(kernelOffset.fX >= 0 && kernelOffset.fX < kernelSize.fWidth); |
| 48 | SkASSERT(kernelOffset.fY >= 0 && kernelOffset.fY < kernelSize.fHeight); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 49 | } |
| 50 | |
robertphillips | ef6a47b | 2016-04-08 08:01:20 -0700 | [diff] [blame] | 51 | sk_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) { |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 60 | if (kernelSize.width() < 1 || kernelSize.height() < 1) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 61 | return nullptr; |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 62 | } |
| 63 | if (gMaxKernelSize / kernelSize.fWidth < kernelSize.fHeight) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 64 | return nullptr; |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 65 | } |
| 66 | if (!kernel) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 67 | return nullptr; |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 68 | } |
sugoi | 1379b87 | 2014-11-17 05:45:55 -0800 | [diff] [blame] | 69 | if ((kernelOffset.fX < 0) || (kernelOffset.fX >= kernelSize.fWidth) || |
| 70 | (kernelOffset.fY < 0) || (kernelOffset.fY >= kernelSize.fHeight)) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 71 | return nullptr; |
sugoi | 1379b87 | 2014-11-17 05:45:55 -0800 | [diff] [blame] | 72 | } |
robertphillips | ef6a47b | 2016-04-08 08:01:20 -0700 | [diff] [blame] | 73 | return sk_sp<SkImageFilter>(new SkMatrixConvolutionImageFilter(kernelSize, kernel, gain, |
| 74 | bias, kernelOffset, |
| 75 | tileMode, convolveAlpha, |
| 76 | std::move(input), cropRect)); |
senorblanco | 3a49520 | 2014-09-29 07:57:20 -0700 | [diff] [blame] | 77 | } |
| 78 | |
reed | 60c9b58 | 2016-04-03 09:11:13 -0700 | [diff] [blame] | 79 | sk_sp<SkFlattenable> SkMatrixConvolutionImageFilter::CreateProc(SkReadBuffer& buffer) { |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 80 | 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)) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 88 | return nullptr; |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 89 | } |
| 90 | SkAutoSTArray<16, SkScalar> kernel(count); |
| 91 | if (!buffer.readScalarArray(kernel.get(), count)) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 92 | return nullptr; |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 93 | } |
| 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(); |
robertphillips | ef6a47b | 2016-04-08 08:01:20 -0700 | [diff] [blame] | 101 | return Make(kernelSize, kernel.get(), gain, bias, kernelOffset, tileMode, |
| 102 | convolveAlpha, common.getInput(0), &common.cropRect()); |
reed | 9fa60da | 2014-08-21 07:59:51 -0700 | [diff] [blame] | 103 | } |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 104 | |
commit-bot@chromium.org | 8b0e8ac | 2014-01-30 18:58:24 +0000 | [diff] [blame] | 105 | void SkMatrixConvolutionImageFilter::flatten(SkWriteBuffer& buffer) const { |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 106 | 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.org | cac5fd5 | 2014-03-10 10:51:58 +0000 | [diff] [blame] | 112 | buffer.writeInt(fKernelOffset.fX); |
| 113 | buffer.writeInt(fKernelOffset.fY); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 114 | buffer.writeInt((int) fTileMode); |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 115 | buffer.writeBool(fConvolveAlpha); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 116 | } |
| 117 | |
| 118 | SkMatrixConvolutionImageFilter::~SkMatrixConvolutionImageFilter() { |
| 119 | delete[] fKernel; |
| 120 | } |
| 121 | |
| 122 | class UncheckedPixelFetcher { |
| 123 | public: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 124 | static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) { |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 125 | return *src.getAddr32(x, y); |
| 126 | } |
| 127 | }; |
| 128 | |
| 129 | class ClampPixelFetcher { |
| 130 | public: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 131 | static inline SkPMColor fetch(const SkBitmap& src, int x, int y, const SkIRect& bounds) { |
bungeman | 62ce030 | 2015-08-28 09:09:32 -0700 | [diff] [blame] | 132 | x = SkTPin(x, bounds.fLeft, bounds.fRight - 1); |
| 133 | y = SkTPin(y, bounds.fTop, bounds.fBottom - 1); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 134 | return *src.getAddr32(x, y); |
| 135 | } |
| 136 | }; |
| 137 | |
| 138 | class RepeatPixelFetcher { |
| 139 | public: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 140 | 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.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 145 | } |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 146 | if (y < bounds.top()) { |
| 147 | y += bounds.height(); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 148 | } |
| 149 | return *src.getAddr32(x, y); |
| 150 | } |
| 151 | }; |
| 152 | |
| 153 | class ClampToBlackPixelFetcher { |
| 154 | public: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 155 | 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.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 157 | return 0; |
| 158 | } else { |
| 159 | return *src.getAddr32(x, y); |
| 160 | } |
| 161 | } |
| 162 | }; |
| 163 | |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 164 | template<class PixelFetcher, bool convolveAlpha> |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 165 | void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src, |
| 166 | SkBitmap* result, |
senorblanco@chromium.org | 8c7372b | 2014-05-02 19:13:11 +0000 | [diff] [blame] | 167 | const SkIRect& r, |
commit-bot@chromium.org | ae761f7 | 2014-02-05 22:32:02 +0000 | [diff] [blame] | 168 | const SkIRect& bounds) const { |
senorblanco@chromium.org | 8c7372b | 2014-05-02 19:13:11 +0000 | [diff] [blame] | 169 | SkIRect rect(r); |
| 170 | if (!rect.intersect(bounds)) { |
senorblanco@chromium.org | 9195743 | 2014-05-01 14:03:41 +0000 | [diff] [blame] | 171 | return; |
| 172 | } |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 173 | for (int y = rect.fTop; y < rect.fBottom; ++y) { |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 174 | SkPMColor* dptr = result->getAddr32(rect.fLeft - bounds.fLeft, y - bounds.fTop); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 175 | 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.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 179 | SkPMColor s = PixelFetcher::fetch(src, |
commit-bot@chromium.org | cac5fd5 | 2014-03-10 10:51:58 +0000 | [diff] [blame] | 180 | x + cx - fKernelOffset.fX, |
| 181 | y + cy - fKernelOffset.fY, |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 182 | bounds); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 183 | SkScalar k = fKernel[cy * fKernelSize.fWidth + cx]; |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 184 | if (convolveAlpha) { |
| 185 | sumA += SkScalarMul(SkIntToScalar(SkGetPackedA32(s)), k); |
| 186 | } |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 187 | sumR += SkScalarMul(SkIntToScalar(SkGetPackedR32(s)), k); |
| 188 | sumG += SkScalarMul(SkIntToScalar(SkGetPackedG32(s)), k); |
| 189 | sumB += SkScalarMul(SkIntToScalar(SkGetPackedB32(s)), k); |
| 190 | } |
| 191 | } |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 192 | int a = convolveAlpha |
| 193 | ? SkClampMax(SkScalarFloorToInt(SkScalarMul(sumA, fGain) + fBias), 255) |
senorblanco@chromium.org | 3bc16c8 | 2012-10-04 17:18:20 +0000 | [diff] [blame] | 194 | : 255; |
senorblanco@chromium.org | cc9471c | 2012-09-20 17:59:49 +0000 | [diff] [blame] | 195 | 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.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 198 | if (!convolveAlpha) { |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 199 | a = SkGetPackedA32(PixelFetcher::fetch(src, x, y, bounds)); |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 200 | *dptr++ = SkPreMultiplyARGB(a, r, g, b); |
| 201 | } else { |
| 202 | *dptr++ = SkPackARGB32(a, r, g, b); |
| 203 | } |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 204 | } |
| 205 | } |
| 206 | } |
| 207 | |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 208 | template<class PixelFetcher> |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 209 | void SkMatrixConvolutionImageFilter::filterPixels(const SkBitmap& src, |
| 210 | SkBitmap* result, |
| 211 | const SkIRect& rect, |
commit-bot@chromium.org | ae761f7 | 2014-02-05 22:32:02 +0000 | [diff] [blame] | 212 | const SkIRect& bounds) const { |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 213 | if (fConvolveAlpha) { |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 214 | filterPixels<PixelFetcher, true>(src, result, rect, bounds); |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 215 | } else { |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 216 | filterPixels<PixelFetcher, false>(src, result, rect, bounds); |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 217 | } |
| 218 | } |
| 219 | |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 220 | void SkMatrixConvolutionImageFilter::filterInteriorPixels(const SkBitmap& src, |
| 221 | SkBitmap* result, |
| 222 | const SkIRect& rect, |
commit-bot@chromium.org | ae761f7 | 2014-02-05 22:32:02 +0000 | [diff] [blame] | 223 | const SkIRect& bounds) const { |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 224 | filterPixels<UncheckedPixelFetcher>(src, result, rect, bounds); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 225 | } |
| 226 | |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 227 | void SkMatrixConvolutionImageFilter::filterBorderPixels(const SkBitmap& src, |
| 228 | SkBitmap* result, |
| 229 | const SkIRect& rect, |
commit-bot@chromium.org | ae761f7 | 2014-02-05 22:32:02 +0000 | [diff] [blame] | 230 | const SkIRect& bounds) const { |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 231 | switch (fTileMode) { |
| 232 | case kClamp_TileMode: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 233 | filterPixels<ClampPixelFetcher>(src, result, rect, bounds); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 234 | break; |
| 235 | case kRepeat_TileMode: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 236 | filterPixels<RepeatPixelFetcher>(src, result, rect, bounds); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 237 | break; |
| 238 | case kClampToBlack_TileMode: |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 239 | filterPixels<ClampToBlackPixelFetcher>(src, result, rect, bounds); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 240 | break; |
| 241 | } |
| 242 | } |
| 243 | |
senorblanco@chromium.org | 377c14a | 2013-02-04 22:57:21 +0000 | [diff] [blame] | 244 | // FIXME: This should be refactored to SkImageFilterUtils for |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 245 | // use by other filters. For now, we assume the input is always |
| 246 | // premultiplied and unpremultiply it |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 247 | static SkBitmap unpremultiply_bitmap(const SkBitmap& src) |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 248 | { |
| 249 | SkAutoLockPixels alp(src); |
| 250 | if (!src.getPixels()) { |
| 251 | return SkBitmap(); |
| 252 | } |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 253 | |
| 254 | const SkImageInfo info = SkImageInfo::MakeN32(src.width(), src.height(), src.alphaType()); |
| 255 | SkBitmap result; |
| 256 | if (!result.tryAllocPixels(info)) { |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 257 | return SkBitmap(); |
| 258 | } |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 259 | SkAutoLockPixels resultLock(result); |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 260 | 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 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 270 | #if SK_SUPPORT_GPU |
senorblanco@chromium.org | 6840076 | 2013-05-24 15:04:07 +0000 | [diff] [blame] | 271 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 272 | static 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 | } |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 285 | #endif |
| 286 | |
| 287 | sk_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.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 294 | } |
| 295 | |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 296 | SkIRect bounds; |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 297 | input = this->applyCropRect(this->mapContext(ctx), input.get(), &inputOffset, &bounds); |
| 298 | if (!input) { |
| 299 | return nullptr; |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 300 | } |
| 301 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 302 | #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 Osman | 615d66d | 2016-12-29 09:18:20 -0500 | [diff] [blame] | 308 | // 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 Osman | 29437eb | 2016-12-27 12:50:37 -0500 | [diff] [blame] | 313 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 314 | 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 | |
bungeman | 06ca8ec | 2016-06-09 08:01:03 -0700 | [diff] [blame] | 321 | sk_sp<GrFragmentProcessor> fp(GrMatrixConvolutionEffect::Make(inputTexture.get(), |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 322 | 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 | |
brianosman | 2a75e5d | 2016-09-22 07:15:37 -0700 | [diff] [blame] | 334 | return DrawWithFP(context, std::move(fp), bounds, ctx.outputProperties()); |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 335 | } |
| 336 | #endif |
| 337 | |
| 338 | SkBitmap inputBM; |
| 339 | |
| 340 | if (!input->getROPixels(&inputBM)) { |
| 341 | return nullptr; |
senorblanco@chromium.org | 8640d50 | 2012-09-25 14:32:42 +0000 | [diff] [blame] | 342 | } |
| 343 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 344 | if (inputBM.colorType() != kN32_SkColorType) { |
| 345 | return nullptr; |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 346 | } |
| 347 | |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 348 | if (!fConvolveAlpha && !inputBM.isOpaque()) { |
| 349 | inputBM = unpremultiply_bitmap(inputBM); |
commit-bot@chromium.org | cd3b15c | 2013-12-04 17:06:49 +0000 | [diff] [blame] | 350 | } |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 351 | |
| 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.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 366 | |
senorblanco@chromium.org | 6776b82 | 2014-01-03 21:48:22 +0000 | [diff] [blame] | 367 | offset->fX = bounds.fLeft; |
| 368 | offset->fY = bounds.fTop; |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 369 | bounds.offset(-inputOffset); |
commit-bot@chromium.org | cac5fd5 | 2014-03-10 10:51:58 +0000 | [diff] [blame] | 370 | SkIRect interior = SkIRect::MakeXYWH(bounds.left() + fKernelOffset.fX, |
| 371 | bounds.top() + fKernelOffset.fY, |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 372 | 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.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 379 | SkIRect right = SkIRect::MakeLTRB(interior.right(), interior.top(), |
senorblanco@chromium.org | 7938bae | 2013-10-18 20:08:14 +0000 | [diff] [blame] | 380 | bounds.right(), interior.bottom()); |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 381 | 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); |
robertphillips | 3e30227 | 2016-04-20 11:48:36 -0700 | [diff] [blame] | 386 | return SkSpecialImage::MakeFromRaster(SkIRect::MakeWH(bounds.width(), bounds.height()), |
robertphillips | dada4dd | 2016-04-13 04:54:36 -0700 | [diff] [blame] | 387 | dst); |
senorblanco@chromium.org | 5faa2dc | 2012-09-18 20:32:34 +0000 | [diff] [blame] | 388 | } |
senorblanco@chromium.org | 3bc16c8 | 2012-10-04 17:18:20 +0000 | [diff] [blame] | 389 | |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 390 | SkIRect SkMatrixConvolutionImageFilter::onFilterNodeBounds(const SkIRect& src, const SkMatrix& ctm, |
| 391 | MapDirection direction) const { |
| 392 | SkIRect dst = src; |
senorblanco | db64af3 | 2015-12-09 10:11:43 -0800 | [diff] [blame] | 393 | int w = fKernelSize.width() - 1, h = fKernelSize.height() - 1; |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 394 | dst.fRight += w; |
| 395 | dst.fBottom += h; |
senorblanco | db64af3 | 2015-12-09 10:11:43 -0800 | [diff] [blame] | 396 | if (kReverse_MapDirection == direction) { |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 397 | dst.offset(-fKernelOffset); |
senorblanco | db64af3 | 2015-12-09 10:11:43 -0800 | [diff] [blame] | 398 | } else { |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 399 | dst.offset(fKernelOffset - SkIPoint::Make(w, h)); |
senorblanco@chromium.org | 0a5c233 | 2014-04-29 15:20:39 +0000 | [diff] [blame] | 400 | } |
senorblanco | e5e7984 | 2016-03-21 14:51:59 -0700 | [diff] [blame] | 401 | return dst; |
senorblanco@chromium.org | 0a5c233 | 2014-04-29 15:20:39 +0000 | [diff] [blame] | 402 | } |
| 403 | |
senorblanco | 6db0a7b | 2016-04-01 16:41:10 -0700 | [diff] [blame] | 404 | bool SkMatrixConvolutionImageFilter::affectsTransparentBlack() const { |
senorblanco | a544eda | 2015-12-07 07:48:34 -0800 | [diff] [blame] | 405 | // Because the kernel is applied in device-space, we have no idea what |
| 406 | // pixels it will affect in object-space. |
senorblanco | 6db0a7b | 2016-04-01 16:41:10 -0700 | [diff] [blame] | 407 | return true; |
senorblanco | a544eda | 2015-12-07 07:48:34 -0800 | [diff] [blame] | 408 | } |
| 409 | |
robertphillips | f3f5bad | 2014-12-19 13:49:15 -0800 | [diff] [blame] | 410 | #ifndef SK_IGNORE_TO_STRING |
| 411 | void 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 |