| /* |
| * Copyright 2007 The Android Open Source Project |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef SkBitmapProcState_DEFINED |
| #define SkBitmapProcState_DEFINED |
| |
| #include "include/core/SkColor.h" |
| #include "include/core/SkMatrix.h" |
| #include "include/core/SkPixmap.h" |
| #include "include/core/SkPoint.h" |
| #include "include/core/SkSamplingOptions.h" |
| #include "include/core/SkScalar.h" |
| #include "include/private/base/SkAssert.h" |
| #include "include/private/base/SkCPUTypes.h" |
| #include "include/private/base/SkDebug.h" |
| #include "include/private/base/SkFixed.h" |
| #include "src/base/SkArenaAlloc.h" |
| #include "src/core/SkMatrixPriv.h" |
| |
| #include <cstddef> |
| #include <cstdint> |
| |
| class SkImage_Base; |
| enum class SkTileMode; |
| |
| typedef SkFixed3232 SkFractionalInt; |
| #define SkScalarToFractionalInt(x) SkScalarToFixed3232(x) |
| #define SkFractionalIntToFixed(x) SkFixed3232ToFixed(x) |
| #define SkFixedToFractionalInt(x) SkFixedToFixed3232(x) |
| #define SkFractionalIntToInt(x) SkFixed3232ToInt(x) |
| |
| struct SkBitmapProcState { |
| SkBitmapProcState(const SkImage_Base* image, SkTileMode tmx, SkTileMode tmy); |
| |
| bool setup(const SkMatrix& inv, SkColor color, const SkSamplingOptions& sampling) { |
| return this->init(inv, color, sampling) |
| && this->chooseProcs(); |
| } |
| |
| typedef void (*ShaderProc32)(const void* ctx, int x, int y, SkPMColor[], int count); |
| |
| typedef void (*MatrixProc)(const SkBitmapProcState&, |
| uint32_t bitmapXY[], |
| int count, |
| int x, int y); |
| |
| typedef void (*SampleProc32)(const SkBitmapProcState&, |
| const uint32_t[], |
| int count, |
| SkPMColor colors[]); |
| |
| const SkImage_Base* fImage; |
| |
| SkPixmap fPixmap; |
| SkMatrix fInvMatrix; // This changes based on tile mode. |
| SkAlpha fPaintAlpha; |
| SkTileMode fTileModeX; |
| SkTileMode fTileModeY; |
| bool fBilerp; |
| |
| SkMatrixPriv::MapXYProc fInvProc; // chooseProcs |
| SkFractionalInt fInvSxFractionalInt; |
| SkFractionalInt fInvKyFractionalInt; |
| |
| SkFixed fFilterOneX; |
| SkFixed fFilterOneY; |
| |
| uint16_t fAlphaScale; // chooseProcs |
| |
| /** Given the byte size of the index buffer to be passed to the matrix proc, |
| return the maximum number of resulting pixels that can be computed |
| (i.e. the number of SkPMColor values to be written by the sample proc). |
| This routine takes into account that filtering and scale-vs-affine |
| affect the amount of buffer space needed. |
| |
| Only valid to call after chooseProcs (setContext) has been called. It is |
| safe to call this inside the shader's shadeSpan() method. |
| */ |
| int maxCountForBufferSize(size_t bufferSize) const; |
| |
| // If a shader proc is present, then the corresponding matrix/sample procs |
| // are ignored |
| ShaderProc32 getShaderProc32() const { return fShaderProc32; } |
| |
| #ifdef SK_DEBUG |
| MatrixProc getMatrixProc() const; |
| #else |
| MatrixProc getMatrixProc() const { return fMatrixProc; } |
| #endif |
| SampleProc32 getSampleProc32() const { return fSampleProc32; } |
| |
| private: |
| enum { |
| kBMStateSize = 136 // found by inspection. if too small, we will call new/delete |
| }; |
| SkSTArenaAlloc<kBMStateSize> fAlloc; |
| |
| ShaderProc32 fShaderProc32; // chooseProcs |
| // These are used if the shaderproc is nullptr |
| MatrixProc fMatrixProc; // chooseProcs |
| SampleProc32 fSampleProc32; // chooseProcs |
| |
| bool init(const SkMatrix& inverse, SkAlpha, const SkSamplingOptions&); |
| bool chooseProcs(); |
| MatrixProc chooseMatrixProc(bool trivial_matrix); |
| ShaderProc32 chooseShaderProc32(); |
| |
| // Return false if we failed to setup for fast translate (e.g. overflow) |
| bool setupForTranslate(); |
| |
| #ifdef SK_DEBUG |
| static void DebugMatrixProc(const SkBitmapProcState&, |
| uint32_t[], int count, int x, int y); |
| #endif |
| }; |
| |
| /* Macros for packing and unpacking pairs of 16bit values in a 32bit uint. |
| Used to allow access to a stream of uint16_t either one at a time, or |
| 2 at a time by unpacking a uint32_t |
| */ |
| #ifdef SK_CPU_BENDIAN |
| #define PACK_TWO_SHORTS(pri, sec) ((pri) << 16 | (sec)) |
| #define UNPACK_PRIMARY_SHORT(packed) ((uint32_t)(packed) >> 16) |
| #define UNPACK_SECONDARY_SHORT(packed) ((packed) & 0xFFFF) |
| #else |
| #define PACK_TWO_SHORTS(pri, sec) ((pri) | ((sec) << 16)) |
| #define UNPACK_PRIMARY_SHORT(packed) ((packed) & 0xFFFF) |
| #define UNPACK_SECONDARY_SHORT(packed) ((uint32_t)(packed) >> 16) |
| #endif |
| |
| #ifdef SK_DEBUG |
| static inline uint32_t pack_two_shorts(U16CPU pri, U16CPU sec) { |
| SkASSERT((uint16_t)pri == pri); |
| SkASSERT((uint16_t)sec == sec); |
| return PACK_TWO_SHORTS(pri, sec); |
| } |
| #else |
| #define pack_two_shorts(pri, sec) PACK_TWO_SHORTS(pri, sec) |
| #endif |
| |
| // Helper class for mapping the middle of pixel (x, y) into SkFractionalInt bitmap space. |
| // Discussion: |
| // Overall, this code takes a point in destination space, and uses the center of the pixel |
| // at (x, y) to determine the sample point in source space. It then adjusts the pixel by different |
| // amounts based in filtering and tiling. |
| // This code can be broken into two main cases based on filtering: |
| // * no filtering (nearest neighbor) - when using nearest neighbor filtering all tile modes reduce |
| // the sampled by one ulp. If a simple point pt lies precisely on XXX.1/2 then it forced down |
| // when positive making 1/2 + 1/2 = .999999 instead of 1.0. |
| // * filtering - in the filtering case, the code calculates the -1/2 shift for starting the |
| // bilerp kernel. There is a twist; there is a big difference between clamp and the other tile |
| // modes. In tile and repeat the matrix has been reduced by an additional 1/width and 1/height |
| // factor. This maps from destination space to [0, 1) (instead of source space) to allow easy |
| // modulo arithmetic. This means that the -1/2 needed by bilerp is actually 1/2 * 1/width for x |
| // and 1/2 * 1/height for y. This is what happens when the poorly named fFilterOne{X|Y} is |
| // divided by two. |
| class SkBitmapProcStateAutoMapper { |
| public: |
| SkBitmapProcStateAutoMapper(const SkBitmapProcState& s, int x, int y, |
| SkPoint* scalarPoint = nullptr) { |
| SkPoint pt; |
| s.fInvProc(s.fInvMatrix, |
| SkIntToScalar(x) + SK_ScalarHalf, |
| SkIntToScalar(y) + SK_ScalarHalf, &pt); |
| |
| SkFixed biasX = 0, biasY = 0; |
| if (s.fBilerp) { |
| biasX = s.fFilterOneX >> 1; |
| biasY = s.fFilterOneY >> 1; |
| } else { |
| // Our rasterizer biases upward. That is a rect from 0.5...1.5 fills pixel 1 and not |
| // pixel 0. To make an image that is mapped 1:1 with device pixels but at a half pixel |
| // offset select every pixel from the src image once we make exact integer pixel sample |
| // values round down not up. Note that a mirror mapping will not have this property. |
| biasX = 1; |
| biasY = 1; |
| } |
| |
| // punt to unsigned for defined underflow behavior |
| fX = (SkFractionalInt)((uint64_t)SkScalarToFractionalInt(pt.x()) - |
| (uint64_t)SkFixedToFractionalInt(biasX)); |
| fY = (SkFractionalInt)((uint64_t)SkScalarToFractionalInt(pt.y()) - |
| (uint64_t)SkFixedToFractionalInt(biasY)); |
| |
| if (scalarPoint) { |
| scalarPoint->set(pt.x() - SkFixedToScalar(biasX), |
| pt.y() - SkFixedToScalar(biasY)); |
| } |
| } |
| |
| SkFractionalInt fractionalIntX() const { return fX; } |
| SkFractionalInt fractionalIntY() const { return fY; } |
| |
| SkFixed fixedX() const { return SkFractionalIntToFixed(fX); } |
| SkFixed fixedY() const { return SkFractionalIntToFixed(fY); } |
| |
| int intX() const { return SkFractionalIntToInt(fX); } |
| int intY() const { return SkFractionalIntToInt(fY); } |
| |
| private: |
| SkFractionalInt fX, fY; |
| }; |
| |
| namespace sktests { |
| // f is the value to pack, max is the largest the value can be. |
| uint32_t pack_clamp(SkFixed f, unsigned max); |
| // As above, but width is the width of the pretend bitmap. |
| uint32_t pack_repeat(SkFixed f, unsigned max, size_t width); |
| uint32_t pack_mirror(SkFixed f, unsigned max, size_t width); |
| } |
| |
| namespace SkOpts { |
| // SkBitmapProcState optimized Shader, Sample, or Matrix procs. |
| extern void (*S32_alpha_D32_filter_DX)(const SkBitmapProcState&, |
| const uint32_t* xy, int count, SkPMColor*); |
| extern void (*S32_alpha_D32_filter_DXDY)(const SkBitmapProcState&, |
| const uint32_t* xy, int count, SkPMColor*); |
| |
| void Init_BitmapProcState(); |
| } // namespace SkOpts |
| |
| #endif |