mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2016 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 "SkBlitter.h" |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 9 | #include "SkBlendModePriv.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 10 | #include "SkColor.h" |
| 11 | #include "SkColorFilter.h" |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 12 | #include "SkFixedAlloc.h" |
Mike Klein | baaf8ad | 2016-09-29 09:04:15 -0400 | [diff] [blame] | 13 | #include "SkOpts.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 14 | #include "SkPM4f.h" |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 15 | #include "SkPM4fPriv.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 16 | #include "SkRasterPipeline.h" |
| 17 | #include "SkShader.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 18 | |
| 19 | |
| 20 | class SkRasterPipelineBlitter : public SkBlitter { |
| 21 | public: |
Mike Klein | fb191da | 2016-11-15 13:20:33 -0500 | [diff] [blame] | 22 | static SkBlitter* Create(const SkPixmap&, const SkPaint&, const SkMatrix& ctm, |
| 23 | SkTBlitterAllocator*); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 24 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 25 | SkRasterPipelineBlitter(SkPixmap dst, SkBlendMode blend, SkPM4f paintColor) |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 26 | : fDst(dst) |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 27 | , fBlend(blend) |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 28 | , fPaintColor(paintColor) |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 29 | , fScratchAlloc(fScratch, sizeof(fScratch)) |
| 30 | , fScratchFallback(&fScratchAlloc) |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 31 | {} |
| 32 | |
| 33 | void blitH (int x, int y, int w) override; |
| 34 | void blitAntiH(int x, int y, const SkAlpha[], const int16_t[]) override; |
| 35 | void blitMask (const SkMask&, const SkIRect& clip) override; |
| 36 | |
| 37 | // TODO: The default implementations of the other blits look fine, |
| 38 | // but some of them like blitV could probably benefit from custom |
| 39 | // blits using something like a SkRasterPipeline::runFew() method. |
| 40 | |
| 41 | private: |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 42 | void append_load_d(SkRasterPipeline*) const; |
| 43 | void append_store (SkRasterPipeline*) const; |
| 44 | void append_blend (SkRasterPipeline*) const; |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 45 | void maybe_clamp (SkRasterPipeline*) const; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 46 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 47 | SkPixmap fDst; |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 48 | SkBlendMode fBlend; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 49 | SkPM4f fPaintColor; |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 50 | SkRasterPipeline fShader; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 51 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 52 | // These functions are compiled lazily when first used. |
Mike Klein | af49b19 | 2016-11-15 08:52:04 -0500 | [diff] [blame] | 53 | std::function<void(size_t, size_t, size_t)> fBlitH = nullptr, |
| 54 | fBlitAntiH = nullptr, |
| 55 | fBlitMaskA8 = nullptr, |
| 56 | fBlitMaskLCD16 = nullptr; |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 57 | |
| 58 | // These values are pointed to by the compiled blit functions |
| 59 | // above, which allows us to adjust them from call to call. |
Mike Klein | babd93e | 2016-11-30 16:05:10 -0500 | [diff] [blame^] | 60 | void* fDstPtr = nullptr; |
| 61 | const void* fMaskPtr = nullptr; |
| 62 | float fCurrentCoverage = 0.0f; |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 63 | |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 64 | // Scratch space for shaders and color filters to use. |
| 65 | char fScratch[64]; |
| 66 | SkFixedAlloc fScratchAlloc; |
| 67 | SkFallbackAlloc fScratchFallback; |
| 68 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 69 | typedef SkBlitter INHERITED; |
| 70 | }; |
| 71 | |
| 72 | SkBlitter* SkCreateRasterPipelineBlitter(const SkPixmap& dst, |
| 73 | const SkPaint& paint, |
Mike Klein | fb191da | 2016-11-15 13:20:33 -0500 | [diff] [blame] | 74 | const SkMatrix& ctm, |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 75 | SkTBlitterAllocator* alloc) { |
Mike Klein | fb191da | 2016-11-15 13:20:33 -0500 | [diff] [blame] | 76 | return SkRasterPipelineBlitter::Create(dst, paint, ctm, alloc); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 77 | } |
| 78 | |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 79 | static bool supported(const SkImageInfo& info) { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 80 | switch (info.colorType()) { |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 81 | case kN32_SkColorType: return info.gammaCloseToSRGB(); |
| 82 | case kRGBA_F16_SkColorType: return true; |
| 83 | case kRGB_565_SkColorType: return true; |
| 84 | default: return false; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 85 | } |
| 86 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 87 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 88 | SkBlitter* SkRasterPipelineBlitter::Create(const SkPixmap& dst, |
| 89 | const SkPaint& paint, |
Mike Klein | fb191da | 2016-11-15 13:20:33 -0500 | [diff] [blame] | 90 | const SkMatrix& ctm, |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 91 | SkTBlitterAllocator* alloc) { |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 92 | auto blitter = alloc->createT<SkRasterPipelineBlitter>( |
| 93 | dst, |
| 94 | paint.getBlendMode(), |
| 95 | SkPM4f_from_SkColor(paint.getColor(), dst.colorSpace())); |
| 96 | |
Mike Klein | f98b730 | 2016-11-06 10:18:06 -0500 | [diff] [blame] | 97 | auto earlyOut = [&] { |
herb | bf6d80a | 2016-11-15 06:26:56 -0800 | [diff] [blame] | 98 | alloc->deleteLast(); |
Mike Klein | f98b730 | 2016-11-06 10:18:06 -0500 | [diff] [blame] | 99 | return nullptr; |
| 100 | }; |
| 101 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 102 | SkBlendMode* blend = &blitter->fBlend; |
| 103 | SkPM4f* paintColor = &blitter->fPaintColor; |
| 104 | SkRasterPipeline* pipeline = &blitter->fShader; |
| 105 | |
| 106 | SkShader* shader = paint.getShader(); |
| 107 | SkColorFilter* colorFilter = paint.getColorFilter(); |
| 108 | |
| 109 | // TODO: all temporary |
Mike Klein | b0d10e0 | 2016-11-12 10:29:26 -0500 | [diff] [blame] | 110 | if (!supported(dst.info()) || !SkBlendMode_AppendStages(*blend)) { |
Mike Klein | f98b730 | 2016-11-06 10:18:06 -0500 | [diff] [blame] | 111 | return earlyOut(); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 112 | } |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 113 | |
Mike Klein | b0d10e0 | 2016-11-12 10:29:26 -0500 | [diff] [blame] | 114 | bool is_opaque = paintColor->a() == 1.0f, |
| 115 | is_constant = true; |
Mike Klein | b0d10e0 | 2016-11-12 10:29:26 -0500 | [diff] [blame] | 116 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 117 | if (shader) { |
Mike Klein | 987de5b | 2016-11-17 10:47:39 -0500 | [diff] [blame] | 118 | if (!shader->appendStages(pipeline, dst.colorSpace(), &blitter->fScratchFallback, |
Mike Klein | 7a14734 | 2016-11-29 15:33:39 -0500 | [diff] [blame] | 119 | ctm, paint)) { |
Mike Klein | b0d10e0 | 2016-11-12 10:29:26 -0500 | [diff] [blame] | 120 | return earlyOut(); |
| 121 | } |
Mike Klein | 43c847b | 2016-11-21 13:05:23 -0500 | [diff] [blame] | 122 | if (!is_opaque) { |
Mike Klein | babd93e | 2016-11-30 16:05:10 -0500 | [diff] [blame^] | 123 | pipeline->append(SkRasterPipeline::scale_1_float, |
Mike Klein | 43c847b | 2016-11-21 13:05:23 -0500 | [diff] [blame] | 124 | &paintColor->fVec[SkPM4f::A]); |
| 125 | } |
Mike Klein | fb191da | 2016-11-15 13:20:33 -0500 | [diff] [blame] | 126 | |
| 127 | is_opaque = is_opaque && shader->isOpaque(); |
| 128 | is_constant = shader->isConstant(); |
Mike Klein | 43c847b | 2016-11-21 13:05:23 -0500 | [diff] [blame] | 129 | } else { |
| 130 | pipeline->append(SkRasterPipeline::constant_color, paintColor); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 131 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 132 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 133 | if (colorFilter) { |
Mike Klein | 744908e | 2016-11-11 12:51:36 -0500 | [diff] [blame] | 134 | if (!colorFilter->appendStages(pipeline, dst.colorSpace(), &blitter->fScratchFallback, |
| 135 | is_opaque)) { |
Mike Klein | f98b730 | 2016-11-06 10:18:06 -0500 | [diff] [blame] | 136 | return earlyOut(); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 137 | } |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 138 | is_opaque = is_opaque && (colorFilter->getFlags() & SkColorFilter::kAlphaUnchanged_Flag); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 139 | } |
| 140 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 141 | if (is_constant) { |
| 142 | pipeline->append(SkRasterPipeline::store_f32, &paintColor); |
Mike Klein | c789b61 | 2016-11-30 13:45:06 -0500 | [diff] [blame] | 143 | pipeline->run(0,0, 1); |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 144 | |
| 145 | *pipeline = SkRasterPipeline(); |
| 146 | pipeline->append(SkRasterPipeline::constant_color, paintColor); |
| 147 | |
| 148 | is_opaque = paintColor->a() == 1.0f; |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 149 | } |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 150 | |
mtklein | a4a4488 | 2016-11-04 13:20:07 -0700 | [diff] [blame] | 151 | if (is_opaque && *blend == SkBlendMode::kSrcOver) { |
| 152 | *blend = SkBlendMode::kSrc; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 153 | } |
| 154 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 155 | return blitter; |
| 156 | } |
| 157 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 158 | void SkRasterPipelineBlitter::append_load_d(SkRasterPipeline* p) const { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 159 | SkASSERT(supported(fDst.info())); |
| 160 | |
| 161 | switch (fDst.info().colorType()) { |
Mike Klein | 729b582 | 2016-11-28 18:23:23 -0500 | [diff] [blame] | 162 | case kRGB_565_SkColorType: p->append(SkRasterPipeline::load_565_d, &fDstPtr); break; |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 163 | case kBGRA_8888_SkColorType: |
Mike Klein | 729b582 | 2016-11-28 18:23:23 -0500 | [diff] [blame] | 164 | case kRGBA_8888_SkColorType: p->append(SkRasterPipeline::load_8888_d, &fDstPtr); break; |
| 165 | case kRGBA_F16_SkColorType: p->append(SkRasterPipeline::load_f16_d, &fDstPtr); break; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 166 | default: break; |
| 167 | } |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 168 | if (fDst.info().colorType() == kBGRA_8888_SkColorType) { |
Mike Klein | 729b582 | 2016-11-28 18:23:23 -0500 | [diff] [blame] | 169 | p->append(SkRasterPipeline::swap_rb_d); |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 170 | } |
| 171 | if (fDst.info().gammaCloseToSRGB()) { |
| 172 | p->append(SkRasterPipeline::from_srgb_d); |
| 173 | } |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 174 | } |
| 175 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 176 | void SkRasterPipelineBlitter::append_store(SkRasterPipeline* p) const { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 177 | SkASSERT(supported(fDst.info())); |
| 178 | |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 179 | if (fDst.info().gammaCloseToSRGB()) { |
| 180 | p->append(SkRasterPipeline::to_srgb); |
| 181 | } |
| 182 | if (fDst.info().colorType() == kBGRA_8888_SkColorType) { |
| 183 | p->append(SkRasterPipeline::swap_rb); |
| 184 | } |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 185 | switch (fDst.info().colorType()) { |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 186 | case kRGB_565_SkColorType: p->append(SkRasterPipeline::store_565, &fDstPtr); break; |
| 187 | case kBGRA_8888_SkColorType: |
| 188 | case kRGBA_8888_SkColorType: p->append(SkRasterPipeline::store_8888, &fDstPtr); break; |
| 189 | case kRGBA_F16_SkColorType: p->append(SkRasterPipeline::store_f16, &fDstPtr); break; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 190 | default: break; |
| 191 | } |
| 192 | } |
| 193 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 194 | void SkRasterPipelineBlitter::append_blend(SkRasterPipeline* p) const { |
| 195 | SkAssertResult(SkBlendMode_AppendStages(fBlend, p)); |
| 196 | } |
Mike Klein | e9f74b8 | 2016-10-25 13:31:21 -0400 | [diff] [blame] | 197 | |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 198 | void SkRasterPipelineBlitter::maybe_clamp(SkRasterPipeline* p) const { |
Mike Klein | e03339a | 2016-11-28 13:24:27 -0500 | [diff] [blame] | 199 | if (SkBlendMode_CanOverflow(fBlend)) { |
| 200 | p->append(SkRasterPipeline::clamp_a); |
| 201 | } |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 202 | } |
| 203 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 204 | void SkRasterPipelineBlitter::blitH(int x, int y, int w) { |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 205 | if (!fBlitH) { |
| 206 | SkRasterPipeline p; |
| 207 | p.extend(fShader); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 208 | if (fBlend != SkBlendMode::kSrc) { |
| 209 | this->append_load_d(&p); |
| 210 | this->append_blend(&p); |
| 211 | this->maybe_clamp(&p); |
| 212 | } |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 213 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 214 | fBlitH = p.compile(); |
| 215 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 216 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 217 | fDstPtr = fDst.writable_addr(0,y); |
Mike Klein | af49b19 | 2016-11-15 08:52:04 -0500 | [diff] [blame] | 218 | fBlitH(x,y, w); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 219 | } |
| 220 | |
| 221 | void SkRasterPipelineBlitter::blitAntiH(int x, int y, const SkAlpha aa[], const int16_t runs[]) { |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 222 | if (!fBlitAntiH) { |
| 223 | SkRasterPipeline p; |
| 224 | p.extend(fShader); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 225 | if (fBlend == SkBlendMode::kSrcOver) { |
Mike Klein | babd93e | 2016-11-30 16:05:10 -0500 | [diff] [blame^] | 226 | p.append(SkRasterPipeline::scale_1_float, &fCurrentCoverage); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 227 | this->append_load_d(&p); |
| 228 | this->append_blend(&p); |
| 229 | } else { |
| 230 | this->append_load_d(&p); |
| 231 | this->append_blend(&p); |
Mike Klein | babd93e | 2016-11-30 16:05:10 -0500 | [diff] [blame^] | 232 | p.append(SkRasterPipeline::lerp_1_float, &fCurrentCoverage); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 233 | } |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 234 | this->maybe_clamp(&p); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 235 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 236 | fBlitAntiH = p.compile(); |
| 237 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 238 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 239 | fDstPtr = fDst.writable_addr(0,y); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 240 | for (int16_t run = *runs; run > 0; run = *runs) { |
Mike Klein | aeb7959 | 2016-11-07 09:29:07 -0500 | [diff] [blame] | 241 | switch (*aa) { |
| 242 | case 0x00: break; |
| 243 | case 0xff: this->blitH(x,y,run); break; |
| 244 | default: |
Mike Klein | babd93e | 2016-11-30 16:05:10 -0500 | [diff] [blame^] | 245 | fCurrentCoverage = *aa * (1/255.0f); |
Mike Klein | af49b19 | 2016-11-15 08:52:04 -0500 | [diff] [blame] | 246 | fBlitAntiH(x,y, run); |
Mike Klein | aeb7959 | 2016-11-07 09:29:07 -0500 | [diff] [blame] | 247 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 248 | x += run; |
| 249 | runs += run; |
| 250 | aa += run; |
| 251 | } |
| 252 | } |
| 253 | |
| 254 | void SkRasterPipelineBlitter::blitMask(const SkMask& mask, const SkIRect& clip) { |
| 255 | if (mask.fFormat == SkMask::kBW_Format) { |
| 256 | // TODO: native BW masks? |
| 257 | return INHERITED::blitMask(mask, clip); |
| 258 | } |
| 259 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 260 | if (mask.fFormat == SkMask::kA8_Format && !fBlitMaskA8) { |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 261 | SkRasterPipeline p; |
| 262 | p.extend(fShader); |
Mike Klein | 6686617 | 2016-11-03 12:22:01 -0400 | [diff] [blame] | 263 | if (fBlend == SkBlendMode::kSrcOver) { |
| 264 | p.append(SkRasterPipeline::scale_u8, &fMaskPtr); |
| 265 | this->append_load_d(&p); |
| 266 | this->append_blend(&p); |
| 267 | } else { |
| 268 | this->append_load_d(&p); |
| 269 | this->append_blend(&p); |
| 270 | p.append(SkRasterPipeline::lerp_u8, &fMaskPtr); |
| 271 | } |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 272 | this->maybe_clamp(&p); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 273 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 274 | fBlitMaskA8 = p.compile(); |
| 275 | } |
| 276 | |
| 277 | if (mask.fFormat == SkMask::kLCD16_Format && !fBlitMaskLCD16) { |
| 278 | SkRasterPipeline p; |
| 279 | p.extend(fShader); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 280 | this->append_load_d(&p); |
| 281 | this->append_blend(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 282 | p.append(SkRasterPipeline::lerp_565, &fMaskPtr); |
Mike Klein | 130863e | 2016-10-27 11:29:36 -0400 | [diff] [blame] | 283 | this->maybe_clamp(&p); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame] | 284 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 285 | fBlitMaskLCD16 = p.compile(); |
| 286 | } |
| 287 | |
| 288 | int x = clip.left(); |
| 289 | for (int y = clip.top(); y < clip.bottom(); y++) { |
| 290 | fDstPtr = fDst.writable_addr(0,y); |
| 291 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 292 | switch (mask.fFormat) { |
| 293 | case SkMask::kA8_Format: |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 294 | fMaskPtr = mask.getAddr8(x,y)-x; |
Mike Klein | af49b19 | 2016-11-15 08:52:04 -0500 | [diff] [blame] | 295 | fBlitMaskA8(x,y, clip.width()); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 296 | break; |
| 297 | case SkMask::kLCD16_Format: |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 298 | fMaskPtr = mask.getAddrLCD16(x,y)-x; |
Mike Klein | af49b19 | 2016-11-15 08:52:04 -0500 | [diff] [blame] | 299 | fBlitMaskLCD16(x,y, clip.width()); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 300 | break; |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 301 | default: |
| 302 | // TODO |
| 303 | break; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 304 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 305 | } |
| 306 | } |