Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 /* | 1 /* |
| 2 * Copyright 2013 Google Inc. | 2 * Copyright 2013 Google Inc. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
| 5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
| 6 */ | 6 */ |
| 7 | 7 |
| 8 #include "SkBitmapProcState.h" | 8 #include "SkBitmapProcState.h" |
| 9 #include "SkBitmap.h" | 9 #include "SkBitmap.h" |
| 10 #include "SkColor.h" | 10 #include "SkColor.h" |
| 11 #include "SkColorPriv.h" | 11 #include "SkColorPriv.h" |
| 12 #include "SkUnPreMultiply.h" | 12 #include "SkUnPreMultiply.h" |
| 13 #include "SkShader.h" | 13 #include "SkShader.h" |
| 14 #include "SkRTConf.h" | 14 #include "SkRTConf.h" |
| 15 #include "SkMath.h" | 15 #include "SkMath.h" |
| 16 | 16 |
| 17 void highQualityFilter(const SkBitmapProcState& s, int x, int y, | 17 void highQualityFilter(const SkBitmapProcState& s, int x, int y, |
| 18 SkPMColor* SK_RESTRICT colors, int count) { | 18 SkPMColor* SK_RESTRICT colors, int count) { |
| 19 | 19 |
| 20 const int maxX = s.fBitmap->width() - 1; | 20 const int maxX = s.fBitmap->width() - 1; |
| 21 const int maxY = s.fBitmap->height() - 1; | 21 const int maxY = s.fBitmap->height() - 1; |
| 22 | 22 |
| 23 while (count-- > 0) { | 23 while (count-- > 0) { |
| 24 SkPoint srcPt; | 24 SkPoint srcPt; |
| 25 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), | 25 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), |
| 26 SkFloatToScalar(y + 0.5f), &srcPt); | 26 SkFloatToScalar(y + 0.5f), &srcPt); |
| 27 srcPt.fX -= SK_ScalarHalf; | 27 srcPt.fX -= SK_ScalarHalf; |
| 28 srcPt.fY -= SK_ScalarHalf; | 28 srcPt.fY -= SK_ScalarHalf; |
| 29 | 29 |
| 30 SkFixed weight = 0; | 30 SkScalar weight = 0; |
| 31 SkFixed fr = 0, fg = 0, fb = 0, fa = 0; | 31 SkScalar fr = 0, fg = 0, fb = 0, fa = 0; |
| 32 | 32 |
| 33 int y0 = SkClampMax(sk_float_ceil2int(SkScalarToFloat(srcPt.fY)-s.getBit mapFilter()->width()), maxY); | 33 int y0 = SkClampMax(SkScalarCeilToInt(srcPt.fY-s.getBitmapFilter()->widt h()), maxY); |
| 34 int y1 = SkClampMax(sk_float_floor2int(SkScalarToFloat(srcPt.fY)+s.getBi tmapFilter()->width()), maxY); | 34 int y1 = SkClampMax(SkScalarFloorToInt(srcPt.fY+s.getBitmapFilter()->wid th()), maxY); |
| 35 int x0 = SkClampMax(sk_float_ceil2int(SkScalarToFloat(srcPt.fX)-s.getBit mapFilter()->width()), maxX); | 35 int x0 = SkClampMax(SkScalarCeilToInt(srcPt.fX-s.getBitmapFilter()->widt h()), maxX); |
| 36 int x1 = SkClampMax(sk_float_floor2int(SkScalarToFloat(srcPt.fX)+s.getBi tmapFilter()->width()), maxX); | 36 int x1 = SkClampMax(SkScalarFloorToInt(srcPt.fX+s.getBitmapFilter()->wid th()), maxX); |
| 37 | 37 |
| 38 for (int src_y = y0; src_y <= y1; src_y++) { | 38 for (int src_y = y0; src_y <= y1; src_y++) { |
| 39 SkFixed yweight = s.getBitmapFilter()->lookup((srcPt.fY - src_y)); | 39 SkScalar yweight = s.getBitmapFilter()->lookupScalar((srcPt.fY - src _y)); |
| 40 | 40 |
| 41 for (int src_x = x0; src_x <= x1 ; src_x++) { | 41 for (int src_x = x0; src_x <= x1 ; src_x++) { |
| 42 SkFixed xweight = s.getBitmapFilter()->lookup((srcPt.fX - src_x) ); | 42 SkScalar xweight = s.getBitmapFilter()->lookupScalar((srcPt.fX - src_x)); |
| 43 | 43 |
| 44 SkFixed combined_weight = SkFixedMul(xweight, yweight); | 44 SkScalar combined_weight = SkScalarMul(xweight, yweight); |
| 45 | 45 |
| 46 SkPMColor c = *s.fBitmap->getAddr32(src_x, src_y); | 46 SkPMColor c = *s.fBitmap->getAddr32(src_x, src_y); |
| 47 fr += combined_weight * SkGetPackedR32(c); | 47 fr += combined_weight * SkGetPackedR32(c); |
| 48 fg += combined_weight * SkGetPackedG32(c); | 48 fg += combined_weight * SkGetPackedG32(c); |
| 49 fb += combined_weight * SkGetPackedB32(c); | 49 fb += combined_weight * SkGetPackedB32(c); |
| 50 fa += combined_weight * SkGetPackedA32(c); | 50 fa += combined_weight * SkGetPackedA32(c); |
| 51 weight += combined_weight; | 51 weight += combined_weight; |
| 52 } | 52 } |
| 53 } | 53 } |
| 54 | 54 |
| 55 fr = SkFixedDiv(fr, weight); | 55 fr = SkScalarDiv(fr, weight); |
|
reed1
2013/07/12 14:01:06
All of the SkScalarDiv() calls are correct, but we
| |
| 56 fg = SkFixedDiv(fg, weight); | 56 fg = SkScalarDiv(fg, weight); |
| 57 fb = SkFixedDiv(fb, weight); | 57 fb = SkScalarDiv(fb, weight); |
| 58 fa = SkFixedDiv(fa, weight); | 58 fa = SkScalarDiv(fa, weight); |
| 59 | 59 |
| 60 int a = SkClampMax(SkFixedRoundToInt(fa), 255); | 60 int a = SkClampMax(SkScalarRoundToInt(fa), 255); |
| 61 int r = SkClampMax(SkFixedRoundToInt(fr), a); | 61 int r = SkClampMax(SkScalarRoundToInt(fr), a); |
| 62 int g = SkClampMax(SkFixedRoundToInt(fg), a); | 62 int g = SkClampMax(SkScalarRoundToInt(fg), a); |
| 63 int b = SkClampMax(SkFixedRoundToInt(fb), a); | 63 int b = SkClampMax(SkScalarRoundToInt(fb), a); |
| 64 | 64 |
| 65 *colors++ = SkPackARGB32(a, r, g, b); | 65 *colors++ = SkPackARGB32(a, r, g, b); |
| 66 | 66 |
| 67 x++; | 67 x++; |
| 68 } | 68 } |
| 69 } | 69 } |
| 70 | 70 |
| 71 void highQualityFilter_ScaleOnly(const SkBitmapProcState &s, int x, int y, | 71 void highQualityFilter_ScaleOnly(const SkBitmapProcState &s, int x, int y, |
| 72 SkPMColor *SK_RESTRICT colors, int count) { | 72 SkPMColor *SK_RESTRICT colors, int count) { |
| 73 const int maxX = s.fBitmap->width() - 1; | 73 const int maxX = s.fBitmap->width() - 1; |
| 74 const int maxY = s.fBitmap->height() - 1; | 74 const int maxY = s.fBitmap->height() - 1; |
| 75 | 75 |
| 76 SkPoint srcPt; | 76 SkPoint srcPt; |
| 77 | 77 |
| 78 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), | 78 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), |
| 79 SkFloatToScalar(y + 0.5f), &srcPt); | 79 SkFloatToScalar(y + 0.5f), &srcPt); |
| 80 srcPt.fY -= SK_ScalarHalf; | 80 srcPt.fY -= SK_ScalarHalf; |
| 81 int y0 = SkClampMax(sk_float_ceil2int(SkScalarToFloat(srcPt.fY)-s.getBitmap Filter()->width()), maxY); | 81 int y0 = SkClampMax(SkScalarCeilToInt(srcPt.fY-s.getBitmapFilter()->width() ), maxY); |
| 82 int y1 = SkClampMax(sk_float_floor2int(SkScalarToFloat(srcPt.fY)+s.getBitma pFilter()->width()), maxY); | 82 int y1 = SkClampMax(SkScalarFloorToInt(srcPt.fY+s.getBitmapFilter()->width( )), maxY); |
| 83 | 83 |
| 84 while (count-- > 0) { | 84 while (count-- > 0) { |
| 85 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), | 85 s.fInvProc(*s.fInvMatrix, SkFloatToScalar(x + 0.5f), |
| 86 SkFloatToScalar(y + 0.5f), &srcPt); | 86 SkFloatToScalar(y + 0.5f), &srcPt); |
| 87 srcPt.fX -= SK_ScalarHalf; | 87 srcPt.fX -= SK_ScalarHalf; |
| 88 srcPt.fY -= SK_ScalarHalf; | 88 srcPt.fY -= SK_ScalarHalf; |
| 89 | 89 |
| 90 SkFixed weight = 0; | 90 SkScalar weight = 0; |
| 91 SkFixed fr = 0, fg = 0, fb = 0, fa = 0; | 91 SkScalar fr = 0, fg = 0, fb = 0, fa = 0; |
| 92 | 92 |
| 93 int x0 = SkClampMax(sk_float_ceil2int(SkScalarToFloat(srcPt.fX)-s.getBi tmapFilter()->width()), maxX); | 93 int x0 = SkClampMax(SkScalarCeilToInt(srcPt.fX-s.getBitmapFilter()->wid th()), maxX); |
| 94 int x1 = SkClampMax(sk_float_floor2int(SkScalarToFloat(srcPt.fX)+s.getB itmapFilter()->width()), maxX); | 94 int x1 = SkClampMax(SkScalarFloorToInt(srcPt.fX+s.getBitmapFilter()->wi dth()), maxX); |
| 95 | 95 |
| 96 for (int src_y = y0; src_y <= y1; src_y++) { | 96 for (int src_y = y0; src_y <= y1; src_y++) { |
| 97 SkFixed yweight = s.getBitmapFilter()->lookup((srcPt.fY - src_y)); | 97 SkScalar yweight = s.getBitmapFilter()->lookupScalar((srcPt.fY - sr c_y)); |
| 98 | 98 |
| 99 for (int src_x = x0; src_x <= x1 ; src_x++) { | 99 for (int src_x = x0; src_x <= x1 ; src_x++) { |
| 100 SkFixed xweight = s.getBitmapFilter()->lookup((srcPt.fX - src_x )); | 100 SkScalar xweight = s.getBitmapFilter()->lookupScalar((srcPt.fX - src_x)); |
| 101 | 101 |
| 102 SkFixed combined_weight = SkFixedMul(xweight, yweight); | 102 SkScalar combined_weight = SkScalarMul(xweight, yweight); |
| 103 | 103 |
| 104 SkPMColor c = *s.fBitmap->getAddr32(src_x, src_y); | 104 SkPMColor c = *s.fBitmap->getAddr32(src_x, src_y); |
| 105 fr += combined_weight * SkGetPackedR32(c); | 105 fr += combined_weight * SkGetPackedR32(c); |
| 106 fg += combined_weight * SkGetPackedG32(c); | 106 fg += combined_weight * SkGetPackedG32(c); |
| 107 fb += combined_weight * SkGetPackedB32(c); | 107 fb += combined_weight * SkGetPackedB32(c); |
| 108 fa += combined_weight * SkGetPackedA32(c); | 108 fa += combined_weight * SkGetPackedA32(c); |
| 109 weight += combined_weight; | 109 weight += combined_weight; |
| 110 } | 110 } |
| 111 } | 111 } |
| 112 | 112 |
| 113 fr = SkFixedDiv(fr, weight); | 113 fr = SkScalarDiv(fr, weight); |
| 114 fg = SkFixedDiv(fg, weight); | 114 fg = SkScalarDiv(fg, weight); |
| 115 fb = SkFixedDiv(fb, weight); | 115 fb = SkScalarDiv(fb, weight); |
| 116 fa = SkFixedDiv(fa, weight); | 116 fa = SkScalarDiv(fa, weight); |
| 117 | 117 |
| 118 int a = SkClampMax(SkFixedRoundToInt(fa), 255); | 118 int a = SkClampMax(SkScalarRoundToInt(fa), 255); |
| 119 int r = SkClampMax(SkFixedRoundToInt(fr), a); | 119 int r = SkClampMax(SkScalarRoundToInt(fr), a); |
| 120 int g = SkClampMax(SkFixedRoundToInt(fg), a); | 120 int g = SkClampMax(SkScalarRoundToInt(fg), a); |
| 121 int b = SkClampMax(SkFixedRoundToInt(fb), a); | 121 int b = SkClampMax(SkScalarRoundToInt(fb), a); |
| 122 | 122 |
| 123 *colors++ = SkPackARGB32(a, r, g, b); | 123 *colors++ = SkPackARGB32(a, r, g, b); |
| 124 | 124 |
| 125 x++; | 125 x++; |
| 126 } | 126 } |
| 127 } | 127 } |
| 128 | 128 |
| 129 SK_CONF_DECLARE(const char *, c_bitmapFilter, "bitmap.filter", "mitchell", "Whic h bitmap filter to use [mitchell, sinc, gaussian, triangle, box]"); | 129 SK_CONF_DECLARE(const char *, c_bitmapFilter, "bitmap.filter", "mitchell", "Whic h bitmap filter to use [mitchell, sinc, gaussian, triangle, box]"); |
| 130 | 130 |
| 131 static SkBitmapFilter *allocateBitmapFilter() { | 131 static SkBitmapFilter *allocateBitmapFilter() { |
| 132 if (!strcmp(c_bitmapFilter, "mitchell")) { | 132 if (!strcmp(c_bitmapFilter, "mitchell")) { |
| 133 return SkNEW_ARGS(SkMitchellFilter,(1.f/3.f,1.f/3.f)); | 133 return SkNEW_ARGS(SkMitchellFilter,(1.f/3.f,1.f/3.f)); |
| 134 } else if (!strcmp(c_bitmapFilter, "sinc")) { | 134 } else if (!strcmp(c_bitmapFilter, "sinc")) { |
| 135 return SkNEW_ARGS(SkSincFilter,(3)); | 135 return SkNEW_ARGS(SkSincFilter,(3)); |
| 136 } else if (!strcmp(c_bitmapFilter, "gaussian")) { | 136 } else if (!strcmp(c_bitmapFilter, "gaussian")) { |
| 137 return SkNEW_ARGS(SkGaussianFilter,(2)); | 137 return SkNEW_ARGS(SkGaussianFilter,(2)); |
| 138 } else if (!strcmp(c_bitmapFilter, "triangle")) { | 138 } else if (!strcmp(c_bitmapFilter, "triangle")) { |
| 139 return SkNEW(SkTriangleFilter); | 139 return SkNEW(SkTriangleFilter); |
| 140 } else if (!strcmp(c_bitmapFilter, "box")) { | 140 } else if (!strcmp(c_bitmapFilter, "box")) { |
| 141 return SkNEW(SkBoxFilter); | 141 return SkNEW(SkBoxFilter); |
| 142 } else { | 142 } else { |
| 143 SkASSERT(!!!"Unknown filter type"); | 143 SkASSERT(!!!"Unknown filter type"); |
| 144 } | 144 } |
| 145 | 145 |
| 146 return NULL; | 146 return NULL; |
| 147 } | 147 } |
| 148 | 148 |
| 149 SkBitmapProcState::ShaderProc32 | 149 SkBitmapProcState::ShaderProc32 |
| 150 SkBitmapProcState::chooseBitmapFilterProc(const SkPaint& paint) { | 150 SkBitmapProcState::chooseBitmapFilterProc() { |
| 151 // we need to be requested | 151 |
| 152 uint32_t mask = SkPaint::kFilterBitmap_Flag | 152 if (fFilterQuality != kHQ_BitmapFilter) { |
| 153 | SkPaint::kHighQualityFilterBitmap_Flag | 153 return NULL; |
| 154 ; | 154 } |
| 155 if ((paint.getFlags() & mask) != mask) { | 155 |
| 156 if (fAlphaScale != 256) { | |
| 156 return NULL; | 157 return NULL; |
| 157 } | 158 } |
| 158 | 159 |
| 159 // TODO: consider supporting other configs (e.g. 565, A8) | 160 // TODO: consider supporting other configs (e.g. 565, A8) |
| 160 if (fBitmap->config() != SkBitmap::kARGB_8888_Config) { | 161 if (fBitmap->config() != SkBitmap::kARGB_8888_Config) { |
| 161 return NULL; | 162 return NULL; |
| 162 } | 163 } |
| 163 | 164 |
| 164 // TODO: consider supporting repeat and mirror | 165 // TODO: consider supporting repeat and mirror |
| 165 if (SkShader::kClamp_TileMode != fTileModeX || SkShader::kClamp_TileMode != fTileModeY) { | 166 if (SkShader::kClamp_TileMode != fTileModeX || SkShader::kClamp_TileMode != fTileModeY) { |
| 166 return NULL; | 167 return NULL; |
| 167 } | 168 } |
| 168 | 169 |
| 169 // TODO: support blending inside our procs | |
| 170 if (0xFF != paint.getAlpha()) { | |
| 171 return NULL; | |
| 172 } | |
| 173 | |
| 174 if (fInvType & (SkMatrix::kAffine_Mask | SkMatrix::kScale_Mask)) { | 170 if (fInvType & (SkMatrix::kAffine_Mask | SkMatrix::kScale_Mask)) { |
| 175 fBitmapFilter = allocateBitmapFilter(); | 171 fBitmapFilter = allocateBitmapFilter(); |
| 176 } | 172 } |
| 177 | 173 |
| 178 if (fInvType & SkMatrix::kAffine_Mask) { | 174 if (fInvType & SkMatrix::kAffine_Mask) { |
| 179 return highQualityFilter; | 175 return highQualityFilter; |
| 180 } else if (fInvType & SkMatrix::kScale_Mask) { | 176 } else if (fInvType & SkMatrix::kScale_Mask) { |
| 181 return highQualityFilter_ScaleOnly; | 177 return highQualityFilter_ScaleOnly; |
| 182 } else { | 178 } else { |
| 183 return NULL; | 179 return NULL; |
| 184 } | 180 } |
| 185 } | 181 } |
| 186 | 182 |
| 187 static void divideByWeights(SkFixed *sums, SkFixed *weights, SkBitmap *dst) { | 183 static void divideByWeights(SkScalar *sums, SkScalar *weights, SkBitmap *dst) { |
| 188 for (int y = 0 ; y < dst->height() ; y++) { | 184 for (int y = 0 ; y < dst->height() ; y++) { |
| 189 for (int x = 0 ; x < dst->width() ; x++) { | 185 for (int x = 0 ; x < dst->width() ; x++) { |
| 190 SkFixed fr = SkFixedDiv(sums[4*(y*dst->width() + x) + 0], weights[y* dst->width() + x]); | 186 SkScalar fr = SkScalarDiv(sums[4*(y*dst->width() + x) + 0], weights[ y*dst->width() + x]); |
| 191 SkFixed fg = SkFixedDiv(sums[4*(y*dst->width() + x) + 1], weights[y* dst->width() + x]); | 187 SkScalar fg = SkScalarDiv(sums[4*(y*dst->width() + x) + 1], weights[ y*dst->width() + x]); |
| 192 SkFixed fb = SkFixedDiv(sums[4*(y*dst->width() + x) + 2], weights[y* dst->width() + x]); | 188 SkScalar fb = SkScalarDiv(sums[4*(y*dst->width() + x) + 2], weights[ y*dst->width() + x]); |
| 193 SkFixed fa = SkFixedDiv(sums[4*(y*dst->width() + x) + 3], weights[y* dst->width() + x]); | 189 SkScalar fa = SkScalarDiv(sums[4*(y*dst->width() + x) + 3], weights[ y*dst->width() + x]); |
| 194 int a = SkClampMax(SkFixedRoundToInt(fa), 255); | 190 int a = SkClampMax(SkScalarRoundToInt(fa), 255); |
| 195 int r = SkClampMax(SkFixedRoundToInt(fr), a); | 191 int r = SkClampMax(SkScalarRoundToInt(fr), a); |
| 196 int g = SkClampMax(SkFixedRoundToInt(fg), a); | 192 int g = SkClampMax(SkScalarRoundToInt(fg), a); |
| 197 int b = SkClampMax(SkFixedRoundToInt(fb), a); | 193 int b = SkClampMax(SkScalarRoundToInt(fb), a); |
| 198 | 194 |
| 199 *dst->getAddr32(x,y) = SkPackARGB32(a, r, g, b); | 195 *dst->getAddr32(x,y) = SkPackARGB32(a, r, g, b); |
| 200 } | 196 } |
| 201 } | 197 } |
| 202 } | 198 } |
| 203 | 199 |
| 204 static void upScaleHoriz(const SkBitmap *src, SkBitmap *dst, float scale, SkBitm apFilter *filter) { | 200 static void upScaleHorizTranspose(const SkBitmap *src, SkBitmap *dst, float scal e, SkBitmapFilter *filter) { |
| 205 for (int y = 0 ; y < src->height() ; y++) { | 201 for (int y = 0 ; y < dst->height() ; y++) { |
| 206 for (int x = 0 ; x < dst->width() ; x++) { | 202 for (int x = 0 ; x < dst->width() ; x++) { |
| 207 float sx = (x + 0.5f) / scale - 0.5f; | 203 float sx = (y + 0.5f) / scale - 0.5f; |
| 208 int x0 = SkClampMax(sk_float_ceil2int(sx-filter->width()), src->widt h()-1); | 204 int x0 = SkClampMax(sk_float_ceil2int(sx-filter->width()), src->widt h()-1); |
| 209 int x1 = SkClampMax(sk_float_floor2int(sx+filter->width()), src->wid th()-1); | 205 int x1 = SkClampMax(sk_float_floor2int(sx+filter->width()), src->wid th()-1); |
| 210 | 206 |
| 211 SkFixed total_weight = 0; | 207 SkScalar total_weight = 0; |
| 212 SkFixed fr = 0, fg = 0, fb = 0, fa = 0; | 208 SkScalar fr = 0, fg = 0, fb = 0, fa = 0; |
| 213 | 209 |
| 214 for (int src_x = x0 ; src_x <= x1 ; src_x++) { | 210 for (int src_x = x0 ; src_x <= x1 ; src_x++) { |
| 215 SkFixed weight = filter->lookup(sx - src_x); | 211 SkScalar weight = filter->lookupScalar(sx - src_x); |
| 216 SkPMColor c = *src->getAddr32(src_x,y); | 212 SkPMColor c = *src->getAddr32(src_x, x); |
| 217 fr += weight * SkGetPackedR32(c); | 213 fr += SkScalarMul(weight,SkGetPackedR32(c)); |
| 218 fg += weight * SkGetPackedG32(c); | 214 fg += SkScalarMul(weight,SkGetPackedG32(c)); |
| 219 fb += weight * SkGetPackedB32(c); | 215 fb += SkScalarMul(weight,SkGetPackedB32(c)); |
| 220 fa += weight * SkGetPackedA32(c); | 216 fa += SkScalarMul(weight,SkGetPackedA32(c)); |
| 221 total_weight += weight; | 217 total_weight += weight; |
| 222 } | 218 } |
| 223 fr = SkFixedDiv(fr, total_weight); | 219 fr = SkScalarDiv(fr,total_weight); |
| 224 fg = SkFixedDiv(fg, total_weight); | 220 fg = SkScalarDiv(fg,total_weight); |
| 225 fb = SkFixedDiv(fb, total_weight); | 221 fb = SkScalarDiv(fb,total_weight); |
| 226 fa = SkFixedDiv(fa, total_weight); | 222 fa = SkScalarDiv(fa,total_weight); |
| 227 | 223 |
| 228 int a = SkClampMax(SkFixedRoundToInt(fa), 255); | 224 int a = SkClampMax(SkScalarRoundToInt(fa), 255); |
| 229 int r = SkClampMax(SkFixedRoundToInt(fr), a); | 225 int r = SkClampMax(SkScalarRoundToInt(fr), a); |
| 230 int g = SkClampMax(SkFixedRoundToInt(fg), a); | 226 int g = SkClampMax(SkScalarRoundToInt(fg), a); |
| 231 int b = SkClampMax(SkFixedRoundToInt(fb), a); | 227 int b = SkClampMax(SkScalarRoundToInt(fb), a); |
| 232 | 228 |
| 233 *dst->getAddr32(x,y) = SkPackARGB32(a, r, g, b); | 229 *dst->getAddr32(x,y) = SkPackARGB32(a, r, g, b); |
| 234 } | 230 } |
| 235 } | 231 } |
| 236 } | 232 } |
| 237 | 233 |
| 238 static void downScaleHoriz(const SkBitmap *src, SkBitmap *dst, float scale, SkBi tmapFilter *filter) { | 234 static void downScaleHoriz(const SkBitmap *src, SkBitmap *dst, float scale, SkBi tmapFilter *filter) { |
| 239 SkFixed *sums = SkNEW_ARRAY(SkFixed, dst->width() * dst->height() * 4); | 235 SkScalar *sums = SkNEW_ARRAY(SkScalar, dst->width() * dst->height() * 4); |
| 240 SkFixed *weights = SkNEW_ARRAY(SkFixed, dst->width() * dst->height()); | 236 SkScalar *weights = SkNEW_ARRAY(SkScalar, dst->width() * dst->height()); |
| 241 | 237 |
| 242 SkAutoTDeleteArray<SkFixed> ada1(sums); | 238 SkAutoTDeleteArray<SkScalar> ada1(sums); |
| 243 SkAutoTDeleteArray<SkFixed> ada2(weights); | 239 SkAutoTDeleteArray<SkScalar> ada2(weights); |
| 244 | 240 |
| 245 memset(sums, 0, dst->width() * dst->height() * sizeof(SkFixed) * 4); | 241 memset(sums, 0, dst->width() * dst->height() * sizeof(SkScalar) * 4); |
| 246 memset(weights, 0, dst->width() * dst->height() * sizeof(SkFixed)); | 242 memset(weights, 0, dst->width() * dst->height() * sizeof(SkScalar)); |
| 247 | 243 |
| 248 for (int y = 0 ; y < src->height() ; y++) { | 244 for (int y = 0 ; y < src->height() ; y++) { |
| 249 for (int x = 0 ; x < src->width() ; x++) { | 245 for (int x = 0 ; x < src->width() ; x++) { |
| 250 // splat each source pixel into the destination image | 246 // splat each source pixel into the destination image |
| 251 float dx = (x + 0.5f) * scale - 0.5f; | 247 float dx = (x + 0.5f) * scale - 0.5f; |
| 252 int x0 = SkClampMax(sk_float_ceil2int(dx-filter->width()), dst->widt h()-1); | 248 int x0 = SkClampMax(sk_float_ceil2int(dx-filter->width()), dst->widt h()-1); |
| 253 int x1 = SkClampMax(sk_float_floor2int(dx+filter->width()), dst->wid th()-1); | 249 int x1 = SkClampMax(sk_float_floor2int(dx+filter->width()), dst->wid th()-1); |
| 254 | 250 |
| 255 SkPMColor c = *src->getAddr32(x,y); | 251 SkPMColor c = *src->getAddr32(x,y); |
| 256 | 252 |
| 257 for (int dst_x = x0 ; dst_x <= x1 ; dst_x++) { | 253 for (int dst_x = x0 ; dst_x <= x1 ; dst_x++) { |
| 258 SkFixed weight = filter->lookup(dx - dst_x); | 254 SkScalar weight = filter->lookup(dx - dst_x); |
| 259 sums[4*(y*dst->width() + dst_x) + 0] += weight*SkGetPackedR32(c) ; | 255 sums[4*(y*dst->width() + dst_x) + 0] += weight*SkGetPackedR32(c) ; |
| 260 sums[4*(y*dst->width() + dst_x) + 1] += weight*SkGetPackedG32(c) ; | 256 sums[4*(y*dst->width() + dst_x) + 1] += weight*SkGetPackedG32(c) ; |
| 261 sums[4*(y*dst->width() + dst_x) + 2] += weight*SkGetPackedB32(c) ; | 257 sums[4*(y*dst->width() + dst_x) + 2] += weight*SkGetPackedB32(c) ; |
| 262 sums[4*(y*dst->width() + dst_x) + 3] += weight*SkGetPackedA32(c) ; | 258 sums[4*(y*dst->width() + dst_x) + 3] += weight*SkGetPackedA32(c) ; |
| 263 weights[y*dst->width() + dst_x] += weight; | 259 weights[y*dst->width() + dst_x] += weight; |
| 264 } | 260 } |
| 265 } | 261 } |
| 266 } | 262 } |
| 267 | 263 |
| 268 divideByWeights(sums, weights, dst); | 264 divideByWeights(sums, weights, dst); |
| 269 } | 265 } |
| 270 | 266 |
| 271 static void upScaleVert(const SkBitmap *src, SkBitmap *dst, float scale, SkBitma pFilter *filter) { | 267 static void downScaleVert(const SkBitmap *src, SkBitmap *dst, float scale, SkBit mapFilter *filter) { |
| 272 for (int y = 0 ; y < dst->height() ; y++) { | 268 SkScalar *sums = SkNEW_ARRAY(SkScalar, dst->width() * dst->height() * 4); |
| 273 for (int x = 0 ; x < dst->width() ; x++) { | 269 SkScalar *weights = SkNEW_ARRAY(SkScalar, dst->width() * dst->height()); |
| 274 float sy = (y + 0.5f) / scale - 0.5f; | |
| 275 int y0 = SkClampMax(sk_float_ceil2int(sy-filter->width()), src->heig ht()-1); | |
| 276 int y1 = SkClampMax(sk_float_floor2int(sy+filter->width()), src->hei ght()-1); | |
| 277 | 270 |
| 278 SkFixed total_weight = 0; | 271 SkAutoTDeleteArray<SkScalar> ada1(sums); |
| 279 SkFixed fr = 0, fg = 0, fb = 0, fa = 0; | 272 SkAutoTDeleteArray<SkScalar> ada2(weights); |
| 280 | 273 |
| 281 for (int src_y = y0 ; src_y <= y1 ; src_y++) { | 274 memset(sums, 0, dst->width() * dst->height() * sizeof(SkScalar) * 4); |
| 282 SkFixed weight = filter->lookup(sy - src_y); | 275 memset(weights, 0, dst->width() * dst->height() * sizeof(SkScalar)); |
| 283 SkPMColor c = *src->getAddr32(x,src_y); | |
| 284 fr += weight * SkGetPackedR32(c); | |
| 285 fg += weight * SkGetPackedG32(c); | |
| 286 fb += weight * SkGetPackedB32(c); | |
| 287 fa += weight * SkGetPackedA32(c); | |
| 288 total_weight += weight; | |
| 289 } | |
| 290 fr = SkFixedDiv(fr, total_weight); | |
| 291 fg = SkFixedDiv(fg, total_weight); | |
| 292 fb = SkFixedDiv(fb, total_weight); | |
| 293 fa = SkFixedDiv(fa, total_weight); | |
| 294 | |
| 295 int a = SkClampMax(SkFixedRoundToInt(fa), 255); | |
| 296 int r = SkClampMax(SkFixedRoundToInt(fr), a); | |
| 297 int g = SkClampMax(SkFixedRoundToInt(fg), a); | |
| 298 int b = SkClampMax(SkFixedRoundToInt(fb), a); | |
| 299 | |
| 300 *dst->getAddr32(x,y) = SkPackARGB32(a, r, g, b); | |
| 301 } | |
| 302 } | |
| 303 } | |
| 304 | |
| 305 static void downScaleVert(const SkBitmap *src, SkBitmap *dst, float scale, SkBit mapFilter *filter) { | |
| 306 SkFixed *sums = SkNEW_ARRAY(SkFixed, dst->width() * dst->height() * 4); | |
| 307 SkFixed *weights = SkNEW_ARRAY(SkFixed, dst->width() * dst->height()); | |
| 308 | |
| 309 SkAutoTDeleteArray<SkFixed> ada1(sums); | |
| 310 SkAutoTDeleteArray<SkFixed> ada2(weights); | |
| 311 | |
| 312 memset(sums, 0, dst->width() * dst->height() * sizeof(SkFixed) * 4); | |
| 313 memset(weights, 0, dst->width() * dst->height() * sizeof(SkFixed)); | |
| 314 | 276 |
| 315 for (int y = 0 ; y < src->height() ; y++) { | 277 for (int y = 0 ; y < src->height() ; y++) { |
| 316 for (int x = 0 ; x < src->width() ; x++) { | 278 for (int x = 0 ; x < src->width() ; x++) { |
| 317 // splat each source pixel into the destination image | 279 // splat each source pixel into the destination image |
| 318 float dy = (y + 0.5f) * scale - 0.5f; | 280 float dy = (y + 0.5f) * scale - 0.5f; |
| 319 int y0 = SkClampMax(sk_float_ceil2int(dy-filter->width()), dst->heig ht()-1); | 281 int y0 = SkClampMax(sk_float_ceil2int(dy-filter->width()), dst->heig ht()-1); |
| 320 int y1 = SkClampMax(sk_float_ceil2int(dy+filter->width()), dst->heig ht()-1); | 282 int y1 = SkClampMax(sk_float_ceil2int(dy+filter->width()), dst->heig ht()-1); |
| 321 | 283 |
| 322 SkPMColor c = *src->getAddr32(x,y); | 284 SkPMColor c = *src->getAddr32(x,y); |
| 323 | 285 |
| 324 for (int dst_y = y0 ; dst_y <= y1 ; dst_y++) { | 286 for (int dst_y = y0 ; dst_y <= y1 ; dst_y++) { |
| 325 SkFixed weight = filter->lookup(dy - dst_y); | 287 SkScalar weight = filter->lookupScalar(dy - dst_y); |
| 326 sums[4*(dst_y*dst->width() + x) + 0] += weight*SkGetPackedR32(c) ; | 288 sums[4*(dst_y*dst->width() + x) + 0] += weight*SkGetPackedR32(c) ; |
| 327 sums[4*(dst_y*dst->width() + x) + 1] += weight*SkGetPackedG32(c) ; | 289 sums[4*(dst_y*dst->width() + x) + 1] += weight*SkGetPackedG32(c) ; |
| 328 sums[4*(dst_y*dst->width() + x) + 2] += weight*SkGetPackedB32(c) ; | 290 sums[4*(dst_y*dst->width() + x) + 2] += weight*SkGetPackedB32(c) ; |
| 329 sums[4*(dst_y*dst->width() + x) + 3] += weight*SkGetPackedA32(c) ; | 291 sums[4*(dst_y*dst->width() + x) + 3] += weight*SkGetPackedA32(c) ; |
| 330 weights[dst_y*dst->width() + x] += weight; | 292 weights[dst_y*dst->width() + x] += weight; |
| 331 } | 293 } |
| 332 } | 294 } |
| 333 } | 295 } |
| 334 | 296 |
| 335 divideByWeights(sums, weights, dst); | 297 divideByWeights(sums, weights, dst); |
| 336 } | 298 } |
| 337 | 299 |
| 338 void SkBitmap::scale(SkBitmap *dst) const { | 300 void SkBitmap::scale(SkBitmap *dst) const { |
| 339 | 301 |
| 340 SkBitmap horiz_temp; | 302 SkBitmap horiz_temp; |
| 341 | 303 |
| 342 horiz_temp.setConfig(SkBitmap::kARGB_8888_Config, dst->width(), height()); | 304 horiz_temp.setConfig(SkBitmap::kARGB_8888_Config, height(), dst->width()); |
| 343 horiz_temp.allocPixels(); | 305 horiz_temp.allocPixels(); |
| 344 | 306 |
| 345 SkBitmapFilter *filter = allocateBitmapFilter(); | 307 SkBitmapFilter *filter = allocateBitmapFilter(); |
| 346 | 308 |
| 347 float horiz_scale = float(dst->width()) / width(); | 309 float horiz_scale = float(dst->width()) / width(); |
| 348 | 310 |
| 349 if (horiz_scale == 1) { | 311 if (horiz_scale >= 1) { |
| 350 this->copyPixelsTo(horiz_temp.getPixels(), getSize()); | 312 upScaleHorizTranspose(this, &horiz_temp, horiz_scale, filter); |
| 351 } else if (horiz_scale > 1) { | |
| 352 upScaleHoriz(this, &horiz_temp, horiz_scale, filter); | |
| 353 } else if (horiz_scale < 1) { | 313 } else if (horiz_scale < 1) { |
| 354 downScaleHoriz(this, &horiz_temp, horiz_scale, filter); | 314 downScaleHoriz(this, &horiz_temp, horiz_scale, filter); |
| 355 } | 315 } |
| 356 | 316 |
| 357 float vert_scale = float(dst->height()) / height(); | 317 float vert_scale = float(dst->height()) / height(); |
| 358 | 318 |
| 359 if (vert_scale == 1) { | 319 if (vert_scale >= 1) { |
| 360 horiz_temp.copyPixelsTo(dst->getPixels(), dst->getSize()); | 320 upScaleHorizTranspose(&horiz_temp, dst, vert_scale, filter); |
| 361 } else if (vert_scale > 1) { | |
| 362 upScaleVert(&horiz_temp, dst, vert_scale, filter); | |
| 363 } else if (vert_scale < 1) { | 321 } else if (vert_scale < 1) { |
| 364 downScaleVert(&horiz_temp, dst, vert_scale, filter); | 322 downScaleVert(&horiz_temp, dst, vert_scale, filter); |
| 365 } | 323 } |
| 366 | 324 |
| 367 SkDELETE(filter); | 325 SkDELETE(filter); |
| 368 } | 326 } |
| OLD | NEW |