• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2006 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 "src/core/SkBlurMask.h"
9 
10 #include "include/core/SkColorPriv.h"
11 #include "include/core/SkMath.h"
12 #include "include/private/SkTemplates.h"
13 #include "include/private/SkTo.h"
14 #include "src/core/SkEndian.h"
15 #include "src/core/SkMaskBlurFilter.h"
16 #include "src/core/SkMathPriv.h"
17 
18 // This constant approximates the scaling done in the software path's
19 // "high quality" mode, in SkBlurMask::Blur() (1 / sqrt(3)).
20 // IMHO, it actually should be 1:  we blur "less" than we should do
21 // according to the CSS and canvas specs, simply because Safari does the same.
22 // Firefox used to do the same too, until 4.0 where they fixed it.  So at some
23 // point we should probably get rid of these scaling constants and rebaseline
24 // all the blur tests.
25 static const SkScalar kBLUR_SIGMA_SCALE = 0.57735f;
26 
ConvertRadiusToSigma(SkScalar radius)27 SkScalar SkBlurMask::ConvertRadiusToSigma(SkScalar radius) {
28     return radius > 0 ? kBLUR_SIGMA_SCALE * radius + 0.5f : 0.0f;
29 }
30 
ConvertSigmaToRadius(SkScalar sigma)31 SkScalar SkBlurMask::ConvertSigmaToRadius(SkScalar sigma) {
32     return sigma > 0.5f ? (sigma - 0.5f) / kBLUR_SIGMA_SCALE : 0.0f;
33 }
34 
35 
36 template <typename AlphaIter>
merge_src_with_blur(uint8_t dst[],int dstRB,AlphaIter src,int srcRB,const uint8_t blur[],int blurRB,int sw,int sh)37 static void merge_src_with_blur(uint8_t dst[], int dstRB,
38                                 AlphaIter src, int srcRB,
39                                 const uint8_t blur[], int blurRB,
40                                 int sw, int sh) {
41     dstRB -= sw;
42     blurRB -= sw;
43     while (--sh >= 0) {
44         AlphaIter rowSrc(src);
45         for (int x = sw - 1; x >= 0; --x) {
46             *dst = SkToU8(SkAlphaMul(*blur, SkAlpha255To256(*rowSrc)));
47             ++dst;
48             ++rowSrc;
49             ++blur;
50         }
51         dst += dstRB;
52         src >>= srcRB;
53         blur += blurRB;
54     }
55 }
56 
57 template <typename AlphaIter>
clamp_solid_with_orig(uint8_t dst[],int dstRowBytes,AlphaIter src,int srcRowBytes,int sw,int sh)58 static void clamp_solid_with_orig(uint8_t dst[], int dstRowBytes,
59                                   AlphaIter src, int srcRowBytes,
60                                   int sw, int sh) {
61     int x;
62     while (--sh >= 0) {
63         AlphaIter rowSrc(src);
64         for (x = sw - 1; x >= 0; --x) {
65             int s = *rowSrc;
66             int d = *dst;
67             *dst = SkToU8(s + d - SkMulDiv255Round(s, d));
68             ++dst;
69             ++rowSrc;
70         }
71         dst += dstRowBytes - sw;
72         src >>= srcRowBytes;
73     }
74 }
75 
76 template <typename AlphaIter>
clamp_outer_with_orig(uint8_t dst[],int dstRowBytes,AlphaIter src,int srcRowBytes,int sw,int sh)77 static void clamp_outer_with_orig(uint8_t dst[], int dstRowBytes,
78                                   AlphaIter src, int srcRowBytes,
79                                   int sw, int sh) {
80     int x;
81     while (--sh >= 0) {
82         AlphaIter rowSrc(src);
83         for (x = sw - 1; x >= 0; --x) {
84             int srcValue = *rowSrc;
85             if (srcValue) {
86                 *dst = SkToU8(SkAlphaMul(*dst, SkAlpha255To256(255 - srcValue)));
87             }
88             ++dst;
89             ++rowSrc;
90         }
91         dst += dstRowBytes - sw;
92         src >>= srcRowBytes;
93     }
94 }
95 ///////////////////////////////////////////////////////////////////////////////
96 
97 // we use a local function to wrap the class static method to work around
98 // a bug in gcc98
99 void SkMask_FreeImage(uint8_t* image);
SkMask_FreeImage(uint8_t * image)100 void SkMask_FreeImage(uint8_t* image) {
101     SkMask::FreeImage(image);
102 }
103 
BoxBlur(SkMask * dst,const SkMask & src,SkScalar sigma,SkBlurStyle style,SkIPoint * margin)104 bool SkBlurMask::BoxBlur(SkMask* dst, const SkMask& src, SkScalar sigma, SkBlurStyle style,
105                          SkIPoint* margin) {
106     if (src.fFormat != SkMask::kBW_Format &&
107         src.fFormat != SkMask::kA8_Format &&
108         src.fFormat != SkMask::kARGB32_Format &&
109         src.fFormat != SkMask::kLCD16_Format)
110     {
111         return false;
112     }
113 
114     SkMaskBlurFilter blurFilter{sigma, sigma};
115     if (blurFilter.hasNoBlur()) {
116         // If there is no effective blur most styles will just produce the original mask.
117         // However, kOuter_SkBlurStyle will produce an empty mask.
118         if (style == kOuter_SkBlurStyle) {
119             dst->fImage = nullptr;
120             dst->fBounds = SkIRect::MakeEmpty();
121             dst->fRowBytes = dst->fBounds.width();
122             dst->fFormat = SkMask::kA8_Format;
123             if (margin != nullptr) {
124                 // This filter will disregard the src.fImage completely.
125                 // The margin is actually {-(src.fBounds.width() / 2), -(src.fBounds.height() / 2)}
126                 // but it is not clear if callers will fall over with negative margins.
127                 *margin = SkIPoint{0,0};
128             }
129             return true;
130         }
131         return false;
132     }
133     const SkIPoint border = blurFilter.blur(src, dst);
134     // If src.fImage is null, then this call is only to calculate the border.
135     if (src.fImage != nullptr && dst->fImage == nullptr) {
136         return false;
137     }
138 
139     if (margin != nullptr) {
140         *margin = border;
141     }
142 
143     if (src.fImage == nullptr) {
144         if (style == kInner_SkBlurStyle) {
145             dst->fBounds = src.fBounds; // restore trimmed bounds
146             dst->fRowBytes = dst->fBounds.width();
147         }
148         return true;
149     }
150 
151     switch (style) {
152         case kNormal_SkBlurStyle:
153             break;
154         case kSolid_SkBlurStyle: {
155             auto dstStart = &dst->fImage[border.x() + border.y() * dst->fRowBytes];
156             switch (src.fFormat) {
157                 case SkMask::kBW_Format:
158                     clamp_solid_with_orig(
159                             dstStart, dst->fRowBytes,
160                             SkMask::AlphaIter<SkMask::kBW_Format>(src.fImage, 0), src.fRowBytes,
161                             src.fBounds.width(), src.fBounds.height());
162                     break;
163                 case SkMask::kA8_Format:
164                     clamp_solid_with_orig(
165                             dstStart, dst->fRowBytes,
166                             SkMask::AlphaIter<SkMask::kA8_Format>(src.fImage), src.fRowBytes,
167                             src.fBounds.width(), src.fBounds.height());
168                     break;
169                 case SkMask::kARGB32_Format: {
170                     uint32_t* srcARGB = reinterpret_cast<uint32_t*>(src.fImage);
171                     clamp_solid_with_orig(
172                             dstStart, dst->fRowBytes,
173                             SkMask::AlphaIter<SkMask::kARGB32_Format>(srcARGB), src.fRowBytes,
174                             src.fBounds.width(), src.fBounds.height());
175                 } break;
176                 case SkMask::kLCD16_Format: {
177                     uint16_t* srcLCD = reinterpret_cast<uint16_t*>(src.fImage);
178                     clamp_solid_with_orig(
179                             dstStart, dst->fRowBytes,
180                             SkMask::AlphaIter<SkMask::kLCD16_Format>(srcLCD), src.fRowBytes,
181                             src.fBounds.width(), src.fBounds.height());
182                 } break;
183                 default:
184                     SK_ABORT("Unhandled format.");
185             }
186         } break;
187         case kOuter_SkBlurStyle: {
188             auto dstStart = &dst->fImage[border.x() + border.y() * dst->fRowBytes];
189             switch (src.fFormat) {
190                 case SkMask::kBW_Format:
191                     clamp_outer_with_orig(
192                             dstStart, dst->fRowBytes,
193                             SkMask::AlphaIter<SkMask::kBW_Format>(src.fImage, 0), src.fRowBytes,
194                             src.fBounds.width(), src.fBounds.height());
195                     break;
196                 case SkMask::kA8_Format:
197                     clamp_outer_with_orig(
198                             dstStart, dst->fRowBytes,
199                             SkMask::AlphaIter<SkMask::kA8_Format>(src.fImage), src.fRowBytes,
200                             src.fBounds.width(), src.fBounds.height());
201                     break;
202                 case SkMask::kARGB32_Format: {
203                     uint32_t* srcARGB = reinterpret_cast<uint32_t*>(src.fImage);
204                     clamp_outer_with_orig(
205                             dstStart, dst->fRowBytes,
206                             SkMask::AlphaIter<SkMask::kARGB32_Format>(srcARGB), src.fRowBytes,
207                             src.fBounds.width(), src.fBounds.height());
208                 } break;
209                 case SkMask::kLCD16_Format: {
210                     uint16_t* srcLCD = reinterpret_cast<uint16_t*>(src.fImage);
211                     clamp_outer_with_orig(
212                             dstStart, dst->fRowBytes,
213                             SkMask::AlphaIter<SkMask::kLCD16_Format>(srcLCD), src.fRowBytes,
214                             src.fBounds.width(), src.fBounds.height());
215                 } break;
216                 default:
217                     SK_ABORT("Unhandled format.");
218             }
219         } break;
220         case kInner_SkBlurStyle: {
221             // now we allocate the "real" dst, mirror the size of src
222             SkMask blur = *dst;
223             SkAutoMaskFreeImage autoFreeBlurMask(blur.fImage);
224             dst->fBounds = src.fBounds;
225             dst->fRowBytes = dst->fBounds.width();
226             size_t dstSize = dst->computeImageSize();
227             if (0 == dstSize) {
228                 return false;   // too big to allocate, abort
229             }
230             dst->fImage = SkMask::AllocImage(dstSize);
231             auto blurStart = &blur.fImage[border.x() + border.y() * blur.fRowBytes];
232             switch (src.fFormat) {
233                 case SkMask::kBW_Format:
234                     merge_src_with_blur(
235                             dst->fImage, dst->fRowBytes,
236                             SkMask::AlphaIter<SkMask::kBW_Format>(src.fImage, 0), src.fRowBytes,
237                             blurStart, blur.fRowBytes,
238                             src.fBounds.width(), src.fBounds.height());
239                     break;
240                 case SkMask::kA8_Format:
241                     merge_src_with_blur(
242                             dst->fImage, dst->fRowBytes,
243                             SkMask::AlphaIter<SkMask::kA8_Format>(src.fImage), src.fRowBytes,
244                             blurStart, blur.fRowBytes,
245                             src.fBounds.width(), src.fBounds.height());
246                     break;
247                 case SkMask::kARGB32_Format: {
248                     uint32_t* srcARGB = reinterpret_cast<uint32_t*>(src.fImage);
249                     merge_src_with_blur(
250                             dst->fImage, dst->fRowBytes,
251                             SkMask::AlphaIter<SkMask::kARGB32_Format>(srcARGB), src.fRowBytes,
252                             blurStart, blur.fRowBytes,
253                             src.fBounds.width(), src.fBounds.height());
254                 } break;
255                 case SkMask::kLCD16_Format: {
256                     uint16_t* srcLCD = reinterpret_cast<uint16_t*>(src.fImage);
257                     merge_src_with_blur(
258                             dst->fImage, dst->fRowBytes,
259                             SkMask::AlphaIter<SkMask::kLCD16_Format>(srcLCD), src.fRowBytes,
260                             blurStart, blur.fRowBytes,
261                             src.fBounds.width(), src.fBounds.height());
262                 } break;
263                 default:
264                     SK_ABORT("Unhandled format.");
265             }
266         } break;
267     }
268 
269     return true;
270 }
271 
272 /* Convolving a box with itself three times results in a piecewise
273    quadratic function:
274 
275    0                              x <= -1.5
276    9/8 + 3/2 x + 1/2 x^2   -1.5 < x <= -.5
277    3/4 - x^2                -.5 < x <= .5
278    9/8 - 3/2 x + 1/2 x^2    0.5 < x <= 1.5
279    0                        1.5 < x
280 
281    Mathematica:
282 
283    g[x_] := Piecewise [ {
284      {9/8 + 3/2 x + 1/2 x^2 ,  -1.5 < x <= -.5},
285      {3/4 - x^2             ,   -.5 < x <= .5},
286      {9/8 - 3/2 x + 1/2 x^2 ,   0.5 < x <= 1.5}
287    }, 0]
288 
289    To get the profile curve of the blurred step function at the rectangle
290    edge, we evaluate the indefinite integral, which is piecewise cubic:
291 
292    0                                        x <= -1.5
293    9/16 + 9/8 x + 3/4 x^2 + 1/6 x^3   -1.5 < x <= -0.5
294    1/2 + 3/4 x - 1/3 x^3              -.5 < x <= .5
295    7/16 + 9/8 x - 3/4 x^2 + 1/6 x^3     .5 < x <= 1.5
296    1                                  1.5 < x
297 
298    in Mathematica code:
299 
300    gi[x_] := Piecewise[ {
301      { 0 , x <= -1.5 },
302      { 9/16 + 9/8 x + 3/4 x^2 + 1/6 x^3, -1.5 < x <= -0.5 },
303      { 1/2 + 3/4 x - 1/3 x^3          ,  -.5 < x <= .5},
304      { 7/16 + 9/8 x - 3/4 x^2 + 1/6 x^3,   .5 < x <= 1.5}
305    },1]
306 */
307 
gaussianIntegral(float x)308 static float gaussianIntegral(float x) {
309     if (x > 1.5f) {
310         return 0.0f;
311     }
312     if (x < -1.5f) {
313         return 1.0f;
314     }
315 
316     float x2 = x*x;
317     float x3 = x2*x;
318 
319     if ( x > 0.5f ) {
320         return 0.5625f - (x3 / 6.0f - 3.0f * x2 * 0.25f + 1.125f * x);
321     }
322     if ( x > -0.5f ) {
323         return 0.5f - (0.75f * x - x3 / 3.0f);
324     }
325     return 0.4375f + (-x3 / 6.0f - 3.0f * x2 * 0.25f - 1.125f * x);
326 }
327 
328 /*  ComputeBlurProfile fills in an array of floating
329     point values between 0 and 255 for the profile signature of
330     a blurred half-plane with the given blur radius.  Since we're
331     going to be doing screened multiplications (i.e., 1 - (1-x)(1-y))
332     all the time, we actually fill in the profile pre-inverted
333     (already done 255-x).
334 */
335 
ComputeBlurProfile(uint8_t * profile,int size,SkScalar sigma)336 void SkBlurMask::ComputeBlurProfile(uint8_t* profile, int size, SkScalar sigma) {
337     SkASSERT(SkScalarCeilToInt(6*sigma) == size);
338 
339     int center = size >> 1;
340 
341     float invr = 1.f/(2*sigma);
342 
343     profile[0] = 255;
344     for (int x = 1 ; x < size ; ++x) {
345         float scaled_x = (center - x - .5f) * invr;
346         float gi = gaussianIntegral(scaled_x);
347         profile[x] = 255 - (uint8_t) (255.f * gi);
348     }
349 }
350 
351 // TODO MAYBE: Maintain a profile cache to avoid recomputing this for
352 // commonly used radii.  Consider baking some of the most common blur radii
353 // directly in as static data?
354 
355 // Implementation adapted from Michael Herf's approach:
356 // http://stereopsis.com/shadowrect/
357 
ProfileLookup(const uint8_t * profile,int loc,int blurredWidth,int sharpWidth)358 uint8_t SkBlurMask::ProfileLookup(const uint8_t *profile, int loc,
359                                   int blurredWidth, int sharpWidth) {
360     // how far are we from the original edge?
361     int dx = SkAbs32(((loc << 1) + 1) - blurredWidth) - sharpWidth;
362     int ox = dx >> 1;
363     if (ox < 0) {
364         ox = 0;
365     }
366 
367     return profile[ox];
368 }
369 
ComputeBlurredScanline(uint8_t * pixels,const uint8_t * profile,unsigned int width,SkScalar sigma)370 void SkBlurMask::ComputeBlurredScanline(uint8_t *pixels, const uint8_t *profile,
371                                         unsigned int width, SkScalar sigma) {
372 
373     unsigned int profile_size = SkScalarCeilToInt(6*sigma);
374     SkAutoTMalloc<uint8_t> horizontalScanline(width);
375 
376     unsigned int sw = width - profile_size;
377     // nearest odd number less than the profile size represents the center
378     // of the (2x scaled) profile
379     int center = ( profile_size & ~1 ) - 1;
380 
381     int w = sw - center;
382 
383     for (unsigned int x = 0 ; x < width ; ++x) {
384        if (profile_size <= sw) {
385            pixels[x] = ProfileLookup(profile, x, width, w);
386        } else {
387            float span = float(sw)/(2*sigma);
388            float giX = 1.5f - (x+.5f)/(2*sigma);
389            pixels[x] = (uint8_t) (255 * (gaussianIntegral(giX) - gaussianIntegral(giX + span)));
390        }
391     }
392 }
393 
BlurRect(SkScalar sigma,SkMask * dst,const SkRect & src,SkBlurStyle style,SkIPoint * margin,SkMask::CreateMode createMode)394 bool SkBlurMask::BlurRect(SkScalar sigma, SkMask *dst,
395                           const SkRect &src, SkBlurStyle style,
396                           SkIPoint *margin, SkMask::CreateMode createMode) {
397     int profileSize = SkScalarCeilToInt(6*sigma);
398     if (profileSize <= 0) {
399         return false;   // no blur to compute
400     }
401 
402     int pad = profileSize/2;
403     if (margin) {
404         margin->set( pad, pad );
405     }
406 
407     dst->fBounds.set(SkScalarRoundToInt(src.fLeft - pad),
408                      SkScalarRoundToInt(src.fTop - pad),
409                      SkScalarRoundToInt(src.fRight + pad),
410                      SkScalarRoundToInt(src.fBottom + pad));
411 
412     dst->fRowBytes = dst->fBounds.width();
413     dst->fFormat = SkMask::kA8_Format;
414     dst->fImage = nullptr;
415 
416     int             sw = SkScalarFloorToInt(src.width());
417     int             sh = SkScalarFloorToInt(src.height());
418 
419     if (createMode == SkMask::kJustComputeBounds_CreateMode) {
420         if (style == kInner_SkBlurStyle) {
421             dst->fBounds.set(SkScalarRoundToInt(src.fLeft),
422                              SkScalarRoundToInt(src.fTop),
423                              SkScalarRoundToInt(src.fRight),
424                              SkScalarRoundToInt(src.fBottom)); // restore trimmed bounds
425             dst->fRowBytes = sw;
426         }
427         return true;
428     }
429 
430     SkAutoTMalloc<uint8_t> profile(profileSize);
431 
432     ComputeBlurProfile(profile, profileSize, sigma);
433 
434     size_t dstSize = dst->computeImageSize();
435     if (0 == dstSize) {
436         return false;   // too big to allocate, abort
437     }
438 
439     uint8_t*        dp = SkMask::AllocImage(dstSize);
440 
441     dst->fImage = dp;
442 
443     int dstHeight = dst->fBounds.height();
444     int dstWidth = dst->fBounds.width();
445 
446     uint8_t *outptr = dp;
447 
448     SkAutoTMalloc<uint8_t> horizontalScanline(dstWidth);
449     SkAutoTMalloc<uint8_t> verticalScanline(dstHeight);
450 
451     ComputeBlurredScanline(horizontalScanline, profile, dstWidth, sigma);
452     ComputeBlurredScanline(verticalScanline, profile, dstHeight, sigma);
453 
454     for (int y = 0 ; y < dstHeight ; ++y) {
455         for (int x = 0 ; x < dstWidth ; x++) {
456             unsigned int maskval = SkMulDiv255Round(horizontalScanline[x], verticalScanline[y]);
457             *(outptr++) = maskval;
458         }
459     }
460 
461     if (style == kInner_SkBlurStyle) {
462         // now we allocate the "real" dst, mirror the size of src
463         size_t srcSize = (size_t)(src.width() * src.height());
464         if (0 == srcSize) {
465             return false;   // too big to allocate, abort
466         }
467         dst->fImage = SkMask::AllocImage(srcSize);
468         for (int y = 0 ; y < sh ; y++) {
469             uint8_t *blur_scanline = dp + (y+pad)*dstWidth + pad;
470             uint8_t *inner_scanline = dst->fImage + y*sw;
471             memcpy(inner_scanline, blur_scanline, sw);
472         }
473         SkMask::FreeImage(dp);
474 
475         dst->fBounds.set(SkScalarRoundToInt(src.fLeft),
476                          SkScalarRoundToInt(src.fTop),
477                          SkScalarRoundToInt(src.fRight),
478                          SkScalarRoundToInt(src.fBottom)); // restore trimmed bounds
479         dst->fRowBytes = sw;
480 
481     } else if (style == kOuter_SkBlurStyle) {
482         for (int y = pad ; y < dstHeight-pad ; y++) {
483             uint8_t *dst_scanline = dp + y*dstWidth + pad;
484             memset(dst_scanline, 0, sw);
485         }
486     } else if (style == kSolid_SkBlurStyle) {
487         for (int y = pad ; y < dstHeight-pad ; y++) {
488             uint8_t *dst_scanline = dp + y*dstWidth + pad;
489             memset(dst_scanline, 0xff, sw);
490         }
491     }
492     // normal and solid styles are the same for analytic rect blurs, so don't
493     // need to handle solid specially.
494 
495     return true;
496 }
497 
BlurRRect(SkScalar sigma,SkMask * dst,const SkRRect & src,SkBlurStyle style,SkIPoint * margin,SkMask::CreateMode createMode)498 bool SkBlurMask::BlurRRect(SkScalar sigma, SkMask *dst,
499                            const SkRRect &src, SkBlurStyle style,
500                            SkIPoint *margin, SkMask::CreateMode createMode) {
501     // Temporary for now -- always fail, should cause caller to fall back
502     // to old path.  Plumbing just to land API and parallelize effort.
503 
504     return false;
505 }
506 
507 // The "simple" blur is a direct implementation of separable convolution with a discrete
508 // gaussian kernel.  It's "ground truth" in a sense; too slow to be used, but very
509 // useful for correctness comparisons.
510 
BlurGroundTruth(SkScalar sigma,SkMask * dst,const SkMask & src,SkBlurStyle style,SkIPoint * margin)511 bool SkBlurMask::BlurGroundTruth(SkScalar sigma, SkMask* dst, const SkMask& src,
512                                  SkBlurStyle style, SkIPoint* margin) {
513 
514     if (src.fFormat != SkMask::kA8_Format) {
515         return false;
516     }
517 
518     float variance = sigma * sigma;
519 
520     int windowSize = SkScalarCeilToInt(sigma*6);
521     // round window size up to nearest odd number
522     windowSize |= 1;
523 
524     SkAutoTMalloc<float> gaussWindow(windowSize);
525 
526     int halfWindow = windowSize >> 1;
527 
528     gaussWindow[halfWindow] = 1;
529 
530     float windowSum = 1;
531     for (int x = 1 ; x <= halfWindow ; ++x) {
532         float gaussian = expf(-x*x / (2*variance));
533         gaussWindow[halfWindow + x] = gaussWindow[halfWindow-x] = gaussian;
534         windowSum += 2*gaussian;
535     }
536 
537     // leave the filter un-normalized for now; we will divide by the normalization
538     // sum later;
539 
540     int pad = halfWindow;
541     if (margin) {
542         margin->set( pad, pad );
543     }
544 
545     dst->fBounds = src.fBounds;
546     dst->fBounds.outset(pad, pad);
547 
548     dst->fRowBytes = dst->fBounds.width();
549     dst->fFormat = SkMask::kA8_Format;
550     dst->fImage = nullptr;
551 
552     if (src.fImage) {
553 
554         size_t dstSize = dst->computeImageSize();
555         if (0 == dstSize) {
556             return false;   // too big to allocate, abort
557         }
558 
559         int             srcWidth = src.fBounds.width();
560         int             srcHeight = src.fBounds.height();
561         int             dstWidth = dst->fBounds.width();
562 
563         const uint8_t*  srcPixels = src.fImage;
564         uint8_t*        dstPixels = SkMask::AllocImage(dstSize);
565         SkAutoMaskFreeImage autoFreeDstPixels(dstPixels);
566 
567         // do the actual blur.  First, make a padded copy of the source.
568         // use double pad so we never have to check if we're outside anything
569 
570         int padWidth = srcWidth + 4*pad;
571         int padHeight = srcHeight;
572         int padSize = padWidth * padHeight;
573 
574         SkAutoTMalloc<uint8_t> padPixels(padSize);
575         memset(padPixels, 0, padSize);
576 
577         for (int y = 0 ; y < srcHeight; ++y) {
578             uint8_t* padptr = padPixels + y * padWidth + 2*pad;
579             const uint8_t* srcptr = srcPixels + y * srcWidth;
580             memcpy(padptr, srcptr, srcWidth);
581         }
582 
583         // blur in X, transposing the result into a temporary floating point buffer.
584         // also double-pad the intermediate result so that the second blur doesn't
585         // have to do extra conditionals.
586 
587         int tmpWidth = padHeight + 4*pad;
588         int tmpHeight = padWidth - 2*pad;
589         int tmpSize = tmpWidth * tmpHeight;
590 
591         SkAutoTMalloc<float> tmpImage(tmpSize);
592         memset(tmpImage, 0, tmpSize*sizeof(tmpImage[0]));
593 
594         for (int y = 0 ; y < padHeight ; ++y) {
595             uint8_t *srcScanline = padPixels + y*padWidth;
596             for (int x = pad ; x < padWidth - pad ; ++x) {
597                 float *outPixel = tmpImage + (x-pad)*tmpWidth + y + 2*pad; // transposed output
598                 uint8_t *windowCenter = srcScanline + x;
599                 for (int i = -pad ; i <= pad ; ++i) {
600                     *outPixel += gaussWindow[pad+i]*windowCenter[i];
601                 }
602                 *outPixel /= windowSum;
603             }
604         }
605 
606         // blur in Y; now filling in the actual desired destination.  We have to do
607         // the transpose again; these transposes guarantee that we read memory in
608         // linear order.
609 
610         for (int y = 0 ; y < tmpHeight ; ++y) {
611             float *srcScanline = tmpImage + y*tmpWidth;
612             for (int x = pad ; x < tmpWidth - pad ; ++x) {
613                 float *windowCenter = srcScanline + x;
614                 float finalValue = 0;
615                 for (int i = -pad ; i <= pad ; ++i) {
616                     finalValue += gaussWindow[pad+i]*windowCenter[i];
617                 }
618                 finalValue /= windowSum;
619                 uint8_t *outPixel = dstPixels + (x-pad)*dstWidth + y; // transposed output
620                 int integerPixel = int(finalValue + 0.5f);
621                 *outPixel = SkClampMax( SkClampPos(integerPixel), 255 );
622             }
623         }
624 
625         dst->fImage = dstPixels;
626         switch (style) {
627             case kNormal_SkBlurStyle:
628                 break;
629             case kSolid_SkBlurStyle: {
630                 clamp_solid_with_orig(
631                         dstPixels + pad*dst->fRowBytes + pad, dst->fRowBytes,
632                         SkMask::AlphaIter<SkMask::kA8_Format>(srcPixels), src.fRowBytes,
633                         srcWidth, srcHeight);
634             } break;
635             case kOuter_SkBlurStyle: {
636                 clamp_outer_with_orig(
637                         dstPixels + pad*dst->fRowBytes + pad, dst->fRowBytes,
638                         SkMask::AlphaIter<SkMask::kA8_Format>(srcPixels), src.fRowBytes,
639                         srcWidth, srcHeight);
640             } break;
641             case kInner_SkBlurStyle: {
642                 // now we allocate the "real" dst, mirror the size of src
643                 size_t srcSize = src.computeImageSize();
644                 if (0 == srcSize) {
645                     return false;   // too big to allocate, abort
646                 }
647                 dst->fImage = SkMask::AllocImage(srcSize);
648                 merge_src_with_blur(dst->fImage, src.fRowBytes,
649                     SkMask::AlphaIter<SkMask::kA8_Format>(srcPixels), src.fRowBytes,
650                     dstPixels + pad*dst->fRowBytes + pad,
651                     dst->fRowBytes, srcWidth, srcHeight);
652                 SkMask::FreeImage(dstPixels);
653             } break;
654         }
655         autoFreeDstPixels.release();
656     }
657 
658     if (style == kInner_SkBlurStyle) {
659         dst->fBounds = src.fBounds; // restore trimmed bounds
660         dst->fRowBytes = src.fRowBytes;
661     }
662 
663     return true;
664 }
665