1 /*
2 * Copyright 2011 Google Inc.
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 "include/core/SkImageEncoder.h"
9 #include "include/core/SkPaint.h"
10 #include "include/core/SkShader.h"
11 #include "include/private/SkColorData.h"
12 #include "include/private/SkMacros.h"
13 #include "include/private/SkTPin.h"
14 #include "src/core/SkBitmapCache.h"
15 #include "src/core/SkBitmapProcState.h"
16 #include "src/core/SkMipmap.h"
17 #include "src/core/SkMipmapAccessor.h"
18 #include "src/core/SkOpts.h"
19 #include "src/core/SkResourceCache.h"
20
21 // One-stop-shop shader for,
22 // - nearest-neighbor sampling (_nofilter_),
23 // - clamp tiling in X and Y both (Clamp_),
24 // - with at most a scale and translate matrix (_DX_),
25 // - and no extra alpha applied (_opaque_),
26 // - sampling from 8888 (_S32_) and drawing to 8888 (_S32_).
Clamp_S32_opaque_D32_nofilter_DX_shaderproc(const void * sIn,int x,int y,SkPMColor * dst,int count)27 static void Clamp_S32_opaque_D32_nofilter_DX_shaderproc(const void* sIn, int x, int y,
28 SkPMColor* dst, int count) {
29 const SkBitmapProcState& s = *static_cast<const SkBitmapProcState*>(sIn);
30 SkASSERT(s.fInvMatrix.isScaleTranslate());
31 SkASSERT(s.fAlphaScale == 256);
32
33 const unsigned maxX = s.fPixmap.width() - 1;
34 SkFractionalInt fx;
35 int dstY;
36 {
37 const SkBitmapProcStateAutoMapper mapper(s, x, y);
38 const unsigned maxY = s.fPixmap.height() - 1;
39 dstY = SkTPin<int>(mapper.intY(), 0, maxY);
40 fx = mapper.fractionalIntX();
41 }
42
43 const SkPMColor* src = s.fPixmap.addr32(0, dstY);
44 const SkFractionalInt dx = s.fInvSxFractionalInt;
45
46 // Check if we're safely inside [0...maxX] so no need to clamp each computed index.
47 //
48 if ((uint64_t)SkFractionalIntToInt(fx) <= maxX &&
49 (uint64_t)SkFractionalIntToInt(fx + dx * (count - 1)) <= maxX)
50 {
51 int count4 = count >> 2;
52 for (int i = 0; i < count4; ++i) {
53 SkPMColor src0 = src[SkFractionalIntToInt(fx)]; fx += dx;
54 SkPMColor src1 = src[SkFractionalIntToInt(fx)]; fx += dx;
55 SkPMColor src2 = src[SkFractionalIntToInt(fx)]; fx += dx;
56 SkPMColor src3 = src[SkFractionalIntToInt(fx)]; fx += dx;
57 dst[0] = src0;
58 dst[1] = src1;
59 dst[2] = src2;
60 dst[3] = src3;
61 dst += 4;
62 }
63 for (int i = (count4 << 2); i < count; ++i) {
64 unsigned index = SkFractionalIntToInt(fx);
65 SkASSERT(index <= maxX);
66 *dst++ = src[index];
67 fx += dx;
68 }
69 } else {
70 for (int i = 0; i < count; ++i) {
71 dst[i] = src[SkTPin<int>(SkFractionalIntToInt(fx), 0, maxX)];
72 fx += dx;
73 }
74 }
75 }
76
S32_alpha_D32_nofilter_DX(const SkBitmapProcState & s,const uint32_t * xy,int count,SkPMColor * colors)77 static void S32_alpha_D32_nofilter_DX(const SkBitmapProcState& s,
78 const uint32_t* xy, int count, SkPMColor* colors) {
79 SkASSERT(count > 0 && colors != nullptr);
80 SkASSERT(s.fInvMatrix.isScaleTranslate());
81 SkASSERT(!s.fBilerp);
82 SkASSERT(4 == s.fPixmap.info().bytesPerPixel());
83 SkASSERT(s.fAlphaScale <= 256);
84
85 // xy is a 32-bit y-coordinate, followed by 16-bit x-coordinates.
86 unsigned y = *xy++;
87 SkASSERT(y < (unsigned)s.fPixmap.height());
88
89 auto row = (const SkPMColor*)( (const char*)s.fPixmap.addr() + y * s.fPixmap.rowBytes() );
90
91 if (1 == s.fPixmap.width()) {
92 sk_memset32(colors, SkAlphaMulQ(row[0], s.fAlphaScale), count);
93 return;
94 }
95
96 // Step 4 xs == 2 uint32_t at a time.
97 while (count >= 4) {
98 uint32_t x01 = *xy++,
99 x23 = *xy++;
100
101 SkPMColor p0 = row[UNPACK_PRIMARY_SHORT (x01)];
102 SkPMColor p1 = row[UNPACK_SECONDARY_SHORT(x01)];
103 SkPMColor p2 = row[UNPACK_PRIMARY_SHORT (x23)];
104 SkPMColor p3 = row[UNPACK_SECONDARY_SHORT(x23)];
105
106 *colors++ = SkAlphaMulQ(p0, s.fAlphaScale);
107 *colors++ = SkAlphaMulQ(p1, s.fAlphaScale);
108 *colors++ = SkAlphaMulQ(p2, s.fAlphaScale);
109 *colors++ = SkAlphaMulQ(p3, s.fAlphaScale);
110
111 count -= 4;
112 }
113
114 // Step 1 x == 1 uint16_t at a time.
115 auto x = (const uint16_t*)xy;
116 while (count --> 0) {
117 *colors++ = SkAlphaMulQ(row[*x++], s.fAlphaScale);
118 }
119 }
120
S32_alpha_D32_nofilter_DXDY(const SkBitmapProcState & s,const uint32_t * xy,int count,SkPMColor * colors)121 static void S32_alpha_D32_nofilter_DXDY(const SkBitmapProcState& s,
122 const uint32_t* xy, int count, SkPMColor* colors) {
123 SkASSERT(count > 0 && colors != nullptr);
124 SkASSERT(!s.fBilerp);
125 SkASSERT(4 == s.fPixmap.info().bytesPerPixel());
126 SkASSERT(s.fAlphaScale <= 256);
127
128 auto src = (const char*)s.fPixmap.addr();
129 size_t rb = s.fPixmap.rowBytes();
130
131 while (count --> 0) {
132 uint32_t XY = *xy++,
133 x = XY & 0xffff,
134 y = XY >> 16;
135 SkASSERT(x < (unsigned)s.fPixmap.width ());
136 SkASSERT(y < (unsigned)s.fPixmap.height());
137 *colors++ = ((const SkPMColor*)(src + y*rb))[x];
138 }
139 }
140
SkBitmapProcState(const SkImage_Base * image,SkTileMode tmx,SkTileMode tmy)141 SkBitmapProcState::SkBitmapProcState(const SkImage_Base* image, SkTileMode tmx, SkTileMode tmy)
142 : fImage(image)
143 , fTileModeX(tmx)
144 , fTileModeY(tmy)
145 {}
146
147 // true iff the matrix has a scale and no more than an optional translate.
matrix_only_scale_translate(const SkMatrix & m)148 static bool matrix_only_scale_translate(const SkMatrix& m) {
149 return (m.getType() & ~SkMatrix::kTranslate_Mask) == SkMatrix::kScale_Mask;
150 }
151
152 /**
153 * For the purposes of drawing bitmaps, if a matrix is "almost" translate
154 * go ahead and treat it as if it were, so that subsequent code can go fast.
155 */
just_trans_general(const SkMatrix & matrix)156 static bool just_trans_general(const SkMatrix& matrix) {
157 SkASSERT(matrix_only_scale_translate(matrix));
158
159 const SkScalar tol = SK_Scalar1 / 32768;
160
161 return SkScalarNearlyZero(matrix[SkMatrix::kMScaleX] - SK_Scalar1, tol)
162 && SkScalarNearlyZero(matrix[SkMatrix::kMScaleY] - SK_Scalar1, tol);
163 }
164
165 /**
166 * Determine if the matrix can be treated as integral-only-translate,
167 * for the purpose of filtering.
168 */
just_trans_integral(const SkMatrix & m)169 static bool just_trans_integral(const SkMatrix& m) {
170 static constexpr SkScalar tol = SK_Scalar1 / 256;
171
172 return m.getType() <= SkMatrix::kTranslate_Mask
173 && SkScalarNearlyEqual(m.getTranslateX(), SkScalarRoundToScalar(m.getTranslateX()), tol)
174 && SkScalarNearlyEqual(m.getTranslateY(), SkScalarRoundToScalar(m.getTranslateY()), tol);
175 }
176
valid_for_filtering(unsigned dimension)177 static bool valid_for_filtering(unsigned dimension) {
178 // for filtering, width and height must fit in 14bits, since we use steal
179 // 2 bits from each to store our 4bit subpixel data
180 return (dimension & ~0x3FFF) == 0;
181 }
182
init(const SkMatrix & inv,SkAlpha paintAlpha,const SkSamplingOptions & sampling)183 bool SkBitmapProcState::init(const SkMatrix& inv, SkAlpha paintAlpha,
184 const SkSamplingOptions& sampling) {
185 SkASSERT(!inv.hasPerspective());
186 SkASSERT(SkOpts::S32_alpha_D32_filter_DXDY || inv.isScaleTranslate());
187 SkASSERT(!sampling.useCubic);
188 SkASSERT(sampling.mipmap != SkMipmapMode::kLinear);
189
190 fPixmap.reset();
191 fInvMatrix = inv;
192 fBilerp = false;
193
194 auto* access = SkMipmapAccessor::Make(&fAlloc, (const SkImage*)fImage, inv, sampling.mipmap);
195 if (!access) {
196 return false;
197 }
198 std::tie(fPixmap, fInvMatrix) = access->level();
199
200 fPaintAlpha = paintAlpha;
201 fBilerp = sampling.filter == SkFilterMode::kLinear;
202 SkASSERT(fPixmap.addr());
203
204 bool integral_translate_only = just_trans_integral(fInvMatrix);
205 if (!integral_translate_only) {
206 // Most of the scanline procs deal with "unit" texture coordinates, as this
207 // makes it easy to perform tiling modes (repeat = (x & 0xFFFF)). To generate
208 // those, we divide the matrix by its dimensions here.
209 //
210 // We don't do this if we're either trivial (can ignore the matrix) or clamping
211 // in both X and Y since clamping to width,height is just as easy as to 0xFFFF.
212
213 if (fTileModeX != SkTileMode::kClamp || fTileModeY != SkTileMode::kClamp) {
214 SkMatrixPriv::PostIDiv(&fInvMatrix, fPixmap.width(), fPixmap.height());
215 }
216
217 // Now that all possible changes to the matrix have taken place, check
218 // to see if we're really close to a no-scale matrix. If so, explicitly
219 // set it to be so. Subsequent code may inspect this matrix to choose
220 // a faster path in this case.
221
222 // This code will only execute if the matrix has some scale component;
223 // if it's already pure translate then we won't do this inversion.
224
225 if (matrix_only_scale_translate(fInvMatrix)) {
226 SkMatrix forward;
227 if (fInvMatrix.invert(&forward) && just_trans_general(forward)) {
228 fInvMatrix.setTranslate(-forward.getTranslateX(), -forward.getTranslateY());
229 }
230 }
231
232 // Recompute the flag after matrix adjustments.
233 integral_translate_only = just_trans_integral(fInvMatrix);
234 }
235
236 if (fBilerp &&
237 (!valid_for_filtering(fPixmap.width() | fPixmap.height()) || integral_translate_only)) {
238 fBilerp = false;
239 }
240
241 return true;
242 }
243
244 /*
245 * Analyze filter-quality and matrix, and decide how to implement that.
246 *
247 * In general, we cascade down the request level [ High ... None ]
248 * - for a given level, if we can fulfill it, fine, else
249 * - else we downgrade to the next lower level and try again.
250 * We can always fulfill requests for Low and None
251 * - sometimes we will "ignore" Low and give None, but this is likely a legacy perf hack
252 * and may be removed.
253 */
chooseProcs()254 bool SkBitmapProcState::chooseProcs() {
255 SkASSERT(!fInvMatrix.hasPerspective());
256 SkASSERT(SkOpts::S32_alpha_D32_filter_DXDY || fInvMatrix.isScaleTranslate());
257 SkASSERT(fPixmap.colorType() == kN32_SkColorType);
258 SkASSERT(fPixmap.alphaType() == kPremul_SkAlphaType ||
259 fPixmap.alphaType() == kOpaque_SkAlphaType);
260
261 SkASSERT(fTileModeX != SkTileMode::kDecal);
262
263 fInvProc = SkMatrixPriv::GetMapXYProc(fInvMatrix);
264 fInvSxFractionalInt = SkScalarToFractionalInt(fInvMatrix.getScaleX());
265 fInvKyFractionalInt = SkScalarToFractionalInt(fInvMatrix.getSkewY ());
266
267 fAlphaScale = SkAlpha255To256(fPaintAlpha);
268
269 bool translate_only = (fInvMatrix.getType() & ~SkMatrix::kTranslate_Mask) == 0;
270 fMatrixProc = this->chooseMatrixProc(translate_only);
271 SkASSERT(fMatrixProc);
272
273 if (fInvMatrix.isScaleTranslate()) {
274 fSampleProc32 = fBilerp ? SkOpts::S32_alpha_D32_filter_DX : S32_alpha_D32_nofilter_DX ;
275 } else {
276 fSampleProc32 = fBilerp ? SkOpts::S32_alpha_D32_filter_DXDY : S32_alpha_D32_nofilter_DXDY;
277 }
278 SkASSERT(fSampleProc32);
279
280 // our special-case shaderprocs
281 // TODO: move this one into chooseShaderProc32() or pull all that in here.
282 if (fAlphaScale == 256
283 && !fBilerp
284 && SkTileMode::kClamp == fTileModeX
285 && SkTileMode::kClamp == fTileModeY
286 && fInvMatrix.isScaleTranslate()) {
287 fShaderProc32 = Clamp_S32_opaque_D32_nofilter_DX_shaderproc;
288 } else {
289 fShaderProc32 = this->chooseShaderProc32();
290 }
291
292 return true;
293 }
294
Clamp_S32_D32_nofilter_trans_shaderproc(const void * sIn,int x,int y,SkPMColor * colors,int count)295 static void Clamp_S32_D32_nofilter_trans_shaderproc(const void* sIn,
296 int x, int y,
297 SkPMColor* colors,
298 int count) {
299 const SkBitmapProcState& s = *static_cast<const SkBitmapProcState*>(sIn);
300 SkASSERT(s.fInvMatrix.isTranslate());
301 SkASSERT(count > 0 && colors != nullptr);
302 SkASSERT(!s.fBilerp);
303
304 const int maxX = s.fPixmap.width() - 1;
305 const int maxY = s.fPixmap.height() - 1;
306 int ix = s.fFilterOneX + x;
307 int iy = SkTPin(s.fFilterOneY + y, 0, maxY);
308 const SkPMColor* row = s.fPixmap.addr32(0, iy);
309
310 // clamp to the left
311 if (ix < 0) {
312 int n = std::min(-ix, count);
313 sk_memset32(colors, row[0], n);
314 count -= n;
315 if (0 == count) {
316 return;
317 }
318 colors += n;
319 SkASSERT(-ix == n);
320 ix = 0;
321 }
322 // copy the middle
323 if (ix <= maxX) {
324 int n = std::min(maxX - ix + 1, count);
325 memcpy(colors, row + ix, n * sizeof(SkPMColor));
326 count -= n;
327 if (0 == count) {
328 return;
329 }
330 colors += n;
331 }
332 SkASSERT(count > 0);
333 // clamp to the right
334 sk_memset32(colors, row[maxX], count);
335 }
336
sk_int_mod(int x,int n)337 static inline int sk_int_mod(int x, int n) {
338 SkASSERT(n > 0);
339 if ((unsigned)x >= (unsigned)n) {
340 if (x < 0) {
341 x = n + ~(~x % n);
342 } else {
343 x = x % n;
344 }
345 }
346 return x;
347 }
348
sk_int_mirror(int x,int n)349 static inline int sk_int_mirror(int x, int n) {
350 x = sk_int_mod(x, 2 * n);
351 if (x >= n) {
352 x = n + ~(x - n);
353 }
354 return x;
355 }
356
Repeat_S32_D32_nofilter_trans_shaderproc(const void * sIn,int x,int y,SkPMColor * colors,int count)357 static void Repeat_S32_D32_nofilter_trans_shaderproc(const void* sIn,
358 int x, int y,
359 SkPMColor* colors,
360 int count) {
361 const SkBitmapProcState& s = *static_cast<const SkBitmapProcState*>(sIn);
362 SkASSERT(s.fInvMatrix.isTranslate());
363 SkASSERT(count > 0 && colors != nullptr);
364 SkASSERT(!s.fBilerp);
365
366 const int stopX = s.fPixmap.width();
367 const int stopY = s.fPixmap.height();
368 int ix = s.fFilterOneX + x;
369 int iy = sk_int_mod(s.fFilterOneY + y, stopY);
370 const SkPMColor* row = s.fPixmap.addr32(0, iy);
371
372 ix = sk_int_mod(ix, stopX);
373 for (;;) {
374 int n = std::min(stopX - ix, count);
375 memcpy(colors, row + ix, n * sizeof(SkPMColor));
376 count -= n;
377 if (0 == count) {
378 return;
379 }
380 colors += n;
381 ix = 0;
382 }
383 }
384
filter_32_alpha(unsigned t,SkPMColor color0,SkPMColor color1,SkPMColor * dstColor,unsigned alphaScale)385 static inline void filter_32_alpha(unsigned t,
386 SkPMColor color0,
387 SkPMColor color1,
388 SkPMColor* dstColor,
389 unsigned alphaScale) {
390 SkASSERT((unsigned)t <= 0xF);
391 SkASSERT(alphaScale <= 256);
392
393 const uint32_t mask = 0xFF00FF;
394
395 int scale = 256 - 16*t;
396 uint32_t lo = (color0 & mask) * scale;
397 uint32_t hi = ((color0 >> 8) & mask) * scale;
398
399 scale = 16*t;
400 lo += (color1 & mask) * scale;
401 hi += ((color1 >> 8) & mask) * scale;
402
403 // TODO: if (alphaScale < 256) ...
404 lo = ((lo >> 8) & mask) * alphaScale;
405 hi = ((hi >> 8) & mask) * alphaScale;
406
407 *dstColor = ((lo >> 8) & mask) | (hi & ~mask);
408 }
409
S32_D32_constX_shaderproc(const void * sIn,int x,int y,SkPMColor * colors,int count)410 static void S32_D32_constX_shaderproc(const void* sIn,
411 int x, int y,
412 SkPMColor* colors,
413 int count) {
414 const SkBitmapProcState& s = *static_cast<const SkBitmapProcState*>(sIn);
415 SkASSERT(s.fInvMatrix.isScaleTranslate());
416 SkASSERT(count > 0 && colors != nullptr);
417 SkASSERT(1 == s.fPixmap.width());
418
419 int iY0;
420 int iY1 SK_INIT_TO_AVOID_WARNING;
421 int iSubY SK_INIT_TO_AVOID_WARNING;
422
423 if (s.fBilerp) {
424 SkBitmapProcState::MatrixProc mproc = s.getMatrixProc();
425 uint32_t xy[2];
426
427 mproc(s, xy, 1, x, y);
428
429 iY0 = xy[0] >> 18;
430 iY1 = xy[0] & 0x3FFF;
431 iSubY = (xy[0] >> 14) & 0xF;
432 } else {
433 int yTemp;
434
435 if (s.fInvMatrix.isTranslate()) {
436 yTemp = s.fFilterOneY + y;
437 } else{
438 const SkBitmapProcStateAutoMapper mapper(s, x, y);
439
440 // When the matrix has a scale component the setup code in
441 // chooseProcs multiples the inverse matrix by the inverse of the
442 // bitmap's width and height. Since this method is going to do
443 // its own tiling and sampling we need to undo that here.
444 if (SkTileMode::kClamp != s.fTileModeX || SkTileMode::kClamp != s.fTileModeY) {
445 yTemp = SkFractionalIntToInt(mapper.fractionalIntY() * s.fPixmap.height());
446 } else {
447 yTemp = mapper.intY();
448 }
449 }
450
451 const int stopY = s.fPixmap.height();
452 switch (s.fTileModeY) {
453 case SkTileMode::kClamp:
454 iY0 = SkTPin(yTemp, 0, stopY-1);
455 break;
456 case SkTileMode::kRepeat:
457 iY0 = sk_int_mod(yTemp, stopY);
458 break;
459 case SkTileMode::kMirror:
460 default:
461 iY0 = sk_int_mirror(yTemp, stopY);
462 break;
463 }
464
465 #ifdef SK_DEBUG
466 {
467 const SkBitmapProcStateAutoMapper mapper(s, x, y);
468 int iY2;
469
470 if (!s.fInvMatrix.isTranslate() &&
471 (SkTileMode::kClamp != s.fTileModeX || SkTileMode::kClamp != s.fTileModeY)) {
472 iY2 = SkFractionalIntToInt(mapper.fractionalIntY() * s.fPixmap.height());
473 } else {
474 iY2 = mapper.intY();
475 }
476
477 switch (s.fTileModeY) {
478 case SkTileMode::kClamp:
479 iY2 = SkTPin(iY2, 0, stopY-1);
480 break;
481 case SkTileMode::kRepeat:
482 iY2 = sk_int_mod(iY2, stopY);
483 break;
484 case SkTileMode::kMirror:
485 default:
486 iY2 = sk_int_mirror(iY2, stopY);
487 break;
488 }
489
490 SkASSERT(iY0 == iY2);
491 }
492 #endif
493 }
494
495 const SkPMColor* row0 = s.fPixmap.addr32(0, iY0);
496 SkPMColor color;
497
498 if (s.fBilerp) {
499 const SkPMColor* row1 = s.fPixmap.addr32(0, iY1);
500 filter_32_alpha(iSubY, *row0, *row1, &color, s.fAlphaScale);
501 } else {
502 if (s.fAlphaScale < 256) {
503 color = SkAlphaMulQ(*row0, s.fAlphaScale);
504 } else {
505 color = *row0;
506 }
507 }
508
509 sk_memset32(colors, color, count);
510 }
511
DoNothing_shaderproc(const void *,int x,int y,SkPMColor * colors,int count)512 static void DoNothing_shaderproc(const void*, int x, int y,
513 SkPMColor* colors, int count) {
514 // if we get called, the matrix is too tricky, so we just draw nothing
515 sk_memset32(colors, 0, count);
516 }
517
setupForTranslate()518 bool SkBitmapProcState::setupForTranslate() {
519 SkPoint pt;
520 const SkBitmapProcStateAutoMapper mapper(*this, 0, 0, &pt);
521
522 /*
523 * if the translate is larger than our ints, we can get random results, or
524 * worse, we might get 0x80000000, which wreaks havoc on us, since we can't
525 * negate it.
526 */
527 const SkScalar too_big = SkIntToScalar(1 << 30);
528 if (SkScalarAbs(pt.fX) > too_big || SkScalarAbs(pt.fY) > too_big) {
529 return false;
530 }
531
532 // Since we know we're not filtered, we re-purpose these fields allow
533 // us to go from device -> src coordinates w/ just an integer add,
534 // rather than running through the inverse-matrix
535 fFilterOneX = mapper.intX();
536 fFilterOneY = mapper.intY();
537
538 return true;
539 }
540
chooseShaderProc32()541 SkBitmapProcState::ShaderProc32 SkBitmapProcState::chooseShaderProc32() {
542
543 if (kN32_SkColorType != fPixmap.colorType()) {
544 return nullptr;
545 }
546
547 if (1 == fPixmap.width() && fInvMatrix.isScaleTranslate()) {
548 if (!fBilerp && fInvMatrix.isTranslate() && !this->setupForTranslate()) {
549 return DoNothing_shaderproc;
550 }
551 return S32_D32_constX_shaderproc;
552 }
553
554 if (fAlphaScale < 256) {
555 return nullptr;
556 }
557 if (!fInvMatrix.isTranslate()) {
558 return nullptr;
559 }
560 if (fBilerp) {
561 return nullptr;
562 }
563
564 SkTileMode tx = fTileModeX;
565 SkTileMode ty = fTileModeY;
566
567 if (SkTileMode::kClamp == tx && SkTileMode::kClamp == ty) {
568 if (this->setupForTranslate()) {
569 return Clamp_S32_D32_nofilter_trans_shaderproc;
570 }
571 return DoNothing_shaderproc;
572 }
573 if (SkTileMode::kRepeat == tx && SkTileMode::kRepeat == ty) {
574 if (this->setupForTranslate()) {
575 return Repeat_S32_D32_nofilter_trans_shaderproc;
576 }
577 return DoNothing_shaderproc;
578 }
579 return nullptr;
580 }
581
582 #ifdef SK_DEBUG
583
check_scale_nofilter(uint32_t bitmapXY[],int count,unsigned mx,unsigned my)584 static void check_scale_nofilter(uint32_t bitmapXY[], int count,
585 unsigned mx, unsigned my) {
586 unsigned y = *bitmapXY++;
587 SkASSERT(y < my);
588
589 const uint16_t* xptr = reinterpret_cast<const uint16_t*>(bitmapXY);
590 for (int i = 0; i < count; ++i) {
591 SkASSERT(xptr[i] < mx);
592 }
593 }
594
check_scale_filter(uint32_t bitmapXY[],int count,unsigned mx,unsigned my)595 static void check_scale_filter(uint32_t bitmapXY[], int count,
596 unsigned mx, unsigned my) {
597 uint32_t YY = *bitmapXY++;
598 unsigned y0 = YY >> 18;
599 unsigned y1 = YY & 0x3FFF;
600 SkASSERT(y0 < my);
601 SkASSERT(y1 < my);
602
603 for (int i = 0; i < count; ++i) {
604 uint32_t XX = bitmapXY[i];
605 unsigned x0 = XX >> 18;
606 unsigned x1 = XX & 0x3FFF;
607 SkASSERT(x0 < mx);
608 SkASSERT(x1 < mx);
609 }
610 }
611
check_affine_nofilter(uint32_t bitmapXY[],int count,unsigned mx,unsigned my)612 static void check_affine_nofilter(uint32_t bitmapXY[], int count, unsigned mx, unsigned my) {
613 for (int i = 0; i < count; ++i) {
614 uint32_t XY = bitmapXY[i];
615 unsigned x = XY & 0xFFFF;
616 unsigned y = XY >> 16;
617 SkASSERT(x < mx);
618 SkASSERT(y < my);
619 }
620 }
621
check_affine_filter(uint32_t bitmapXY[],int count,unsigned mx,unsigned my)622 static void check_affine_filter(uint32_t bitmapXY[], int count, unsigned mx, unsigned my) {
623 for (int i = 0; i < count; ++i) {
624 uint32_t YY = *bitmapXY++;
625 unsigned y0 = YY >> 18;
626 unsigned y1 = YY & 0x3FFF;
627 SkASSERT(y0 < my);
628 SkASSERT(y1 < my);
629
630 uint32_t XX = *bitmapXY++;
631 unsigned x0 = XX >> 18;
632 unsigned x1 = XX & 0x3FFF;
633 SkASSERT(x0 < mx);
634 SkASSERT(x1 < mx);
635 }
636 }
637
DebugMatrixProc(const SkBitmapProcState & state,uint32_t bitmapXY[],int count,int x,int y)638 void SkBitmapProcState::DebugMatrixProc(const SkBitmapProcState& state,
639 uint32_t bitmapXY[], int count,
640 int x, int y) {
641 SkASSERT(bitmapXY);
642 SkASSERT(count > 0);
643
644 state.fMatrixProc(state, bitmapXY, count, x, y);
645
646 void (*proc)(uint32_t bitmapXY[], int count, unsigned mx, unsigned my);
647
648 if (state.fInvMatrix.isScaleTranslate()) {
649 proc = state.fBilerp ? check_scale_filter : check_scale_nofilter;
650 } else {
651 proc = state.fBilerp ? check_affine_filter : check_affine_nofilter;
652 }
653
654 proc(bitmapXY, count, state.fPixmap.width(), state.fPixmap.height());
655 }
656
getMatrixProc() const657 SkBitmapProcState::MatrixProc SkBitmapProcState::getMatrixProc() const {
658 return DebugMatrixProc;
659 }
660
661 #endif
662
663 /*
664 The storage requirements for the different matrix procs are as follows,
665 where each X or Y is 2 bytes, and N is the number of pixels/elements:
666
667 scale/translate nofilter Y(4bytes) + N * X
668 affine/perspective nofilter N * (X Y)
669 scale/translate filter Y Y + N * (X X)
670 affine filter N * (Y Y X X)
671 */
maxCountForBufferSize(size_t bufferSize) const672 int SkBitmapProcState::maxCountForBufferSize(size_t bufferSize) const {
673 int32_t size = static_cast<int32_t>(bufferSize);
674
675 size &= ~3; // only care about 4-byte aligned chunks
676 if (fInvMatrix.isScaleTranslate()) {
677 size -= 4; // the shared Y (or YY) coordinate
678 if (size < 0) {
679 size = 0;
680 }
681 size >>= 1;
682 } else {
683 size >>= 2;
684 }
685
686 if (fBilerp) {
687 size >>= 1;
688 }
689
690 return size;
691 }
692
693