1 /* 2 * Copyright 2012 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 #ifndef SkGradientShaderPriv_DEFINED 9 #define SkGradientShaderPriv_DEFINED 10 11 #include "SkGradientShader.h" 12 13 #include "SkArenaAlloc.h" 14 #include "SkMatrix.h" 15 #include "SkShaderBase.h" 16 #include "SkTArray.h" 17 #include "SkTemplates.h" 18 19 class SkColorSpace; 20 class SkColorSpaceXformer; 21 class SkRasterPipeline; 22 class SkReadBuffer; 23 class SkWriteBuffer; 24 25 class SkGradientShaderBase : public SkShaderBase { 26 public: 27 struct Descriptor { 28 Descriptor() { 29 sk_bzero(this, sizeof(*this)); 30 fTileMode = SkShader::kClamp_TileMode; 31 } 32 33 const SkMatrix* fLocalMatrix; 34 const SkColor4f* fColors; 35 sk_sp<SkColorSpace> fColorSpace; 36 const SkScalar* fPos; 37 int fCount; 38 SkShader::TileMode fTileMode; 39 uint32_t fGradFlags; 40 41 void flatten(SkWriteBuffer&) const; 42 }; 43 44 class DescriptorScope : public Descriptor { 45 public: 46 DescriptorScope() {} 47 48 bool unflatten(SkReadBuffer&); 49 50 // fColors and fPos always point into local memory, so they can be safely mutated 51 // 52 SkColor4f* mutableColors() { return const_cast<SkColor4f*>(fColors); } 53 SkScalar* mutablePos() { return const_cast<SkScalar*>(fPos); } 54 55 private: 56 SkSTArray<16, SkColor4f, true> fColorStorage; 57 SkSTArray<16, SkScalar , true> fPosStorage; 58 SkMatrix fLocalMatrixStorage; 59 }; 60 61 SkGradientShaderBase(const Descriptor& desc, const SkMatrix& ptsToUnit); 62 ~SkGradientShaderBase() override; 63 64 bool isOpaque() const override; 65 66 uint32_t getGradFlags() const { return fGradFlags; } 67 68 const SkMatrix& getGradientMatrix() const { return fPtsToUnit; } 69 70 protected: 71 class GradientShaderBase4fContext; 72 73 SkGradientShaderBase(SkReadBuffer& ); 74 void flatten(SkWriteBuffer&) const override; 75 76 void commonAsAGradient(GradientInfo*) const; 77 78 bool onAsLuminanceColor(SkColor*) const override; 79 80 bool onAppendStages(const StageRec&) const override; 81 82 virtual void appendGradientStages(SkArenaAlloc* alloc, SkRasterPipeline* tPipeline, 83 SkRasterPipeline* postPipeline) const = 0; 84 85 template <typename T, typename... Args> 86 static Context* CheckedMakeContext(SkArenaAlloc* alloc, Args&&... args) { 87 auto* ctx = alloc->make<T>(std::forward<Args>(args)...); 88 if (!ctx->isValid()) { 89 return nullptr; 90 } 91 return ctx; 92 } 93 94 struct AutoXformColors { 95 AutoXformColors(const SkGradientShaderBase&, SkColorSpaceXformer*); 96 97 SkAutoSTMalloc<8, SkColor> fColors; 98 }; 99 100 const SkMatrix fPtsToUnit; 101 TileMode fTileMode; 102 uint8_t fGradFlags; 103 104 public: 105 SkScalar getPos(int i) const { 106 SkASSERT(i < fColorCount); 107 return fOrigPos ? fOrigPos[i] : SkIntToScalar(i) / (fColorCount - 1); 108 } 109 110 SkColor getLegacyColor(int i) const { 111 SkASSERT(i < fColorCount); 112 return fOrigColors4f[i].toSkColor(); 113 } 114 115 SkColor4f* fOrigColors4f; // original colors, as linear floats 116 SkScalar* fOrigPos; // original positions 117 int fColorCount; 118 sk_sp<SkColorSpace> fColorSpace; // color space of gradient stops 119 120 bool colorsAreOpaque() const { return fColorsAreOpaque; } 121 122 TileMode getTileMode() const { return fTileMode; } 123 124 private: 125 // Reserve inline space for up to 4 stops. 126 static constexpr size_t kInlineStopCount = 4; 127 static constexpr size_t kInlineStorageSize = (sizeof(SkColor4f) + sizeof(SkScalar)) 128 * kInlineStopCount; 129 SkAutoSTMalloc<kInlineStorageSize, uint8_t> fStorage; 130 131 bool fColorsAreOpaque; 132 133 typedef SkShaderBase INHERITED; 134 }; 135 136 /////////////////////////////////////////////////////////////////////////////// 137 138 struct SkColor4fXformer { 139 SkColor4fXformer(const SkColor4f* colors, int colorCount, SkColorSpace* src, SkColorSpace* dst); 140 141 const SkColor4f* fColors; 142 SkSTArray<4, SkColor4f, true> fStorage; 143 }; 144 145 #endif 146