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 | baaf8ad | 2016-09-29 09:04:15 -0400 | [diff] [blame] | 12 | #include "SkOpts.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 13 | #include "SkPM4f.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 14 | #include "SkRasterPipeline.h" |
| 15 | #include "SkShader.h" |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 16 | #include "SkXfermode.h" |
| 17 | |
| 18 | |
| 19 | class SkRasterPipelineBlitter : public SkBlitter { |
| 20 | public: |
| 21 | static SkBlitter* Create(const SkPixmap&, const SkPaint&, SkTBlitterAllocator*); |
| 22 | |
| 23 | SkRasterPipelineBlitter(SkPixmap dst, |
| 24 | SkRasterPipeline shader, |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 25 | SkBlendMode blend, |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 26 | SkPM4f paintColor) |
| 27 | : fDst(dst) |
| 28 | , fShader(shader) |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 29 | , fBlend(blend) |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 30 | , fPaintColor(paintColor) |
| 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; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 45 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 46 | SkPixmap fDst; |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 47 | SkRasterPipeline fShader; |
| 48 | SkBlendMode fBlend; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 49 | SkPM4f fPaintColor; |
| 50 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 51 | // These functions are compiled lazily when first used. |
| 52 | std::function<void(size_t, size_t)> fBlitH = nullptr, |
| 53 | fBlitAntiH = nullptr, |
| 54 | fBlitMaskA8 = nullptr, |
| 55 | fBlitMaskLCD16 = nullptr; |
| 56 | |
| 57 | // These values are pointed to by the compiled blit functions |
| 58 | // above, which allows us to adjust them from call to call. |
| 59 | void* fDstPtr = nullptr; |
| 60 | const void* fMaskPtr = nullptr; |
| 61 | float fConstantCoverage = 0.0f; |
| 62 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 63 | typedef SkBlitter INHERITED; |
| 64 | }; |
| 65 | |
| 66 | SkBlitter* SkCreateRasterPipelineBlitter(const SkPixmap& dst, |
| 67 | const SkPaint& paint, |
| 68 | SkTBlitterAllocator* alloc) { |
| 69 | return SkRasterPipelineBlitter::Create(dst, paint, alloc); |
| 70 | } |
| 71 | |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 72 | static bool supported(const SkImageInfo& info) { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 73 | switch (info.colorType()) { |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 74 | case kN32_SkColorType: return info.gammaCloseToSRGB(); |
| 75 | case kRGBA_F16_SkColorType: return true; |
| 76 | case kRGB_565_SkColorType: return true; |
| 77 | default: return false; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 78 | } |
| 79 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 80 | |
| 81 | template <typename Effect> |
| 82 | static bool append_effect_stages(const Effect* effect, SkRasterPipeline* pipeline) { |
| 83 | return !effect || effect->appendStages(pipeline); |
| 84 | } |
| 85 | |
| 86 | |
| 87 | SkBlitter* SkRasterPipelineBlitter::Create(const SkPixmap& dst, |
| 88 | const SkPaint& paint, |
| 89 | SkTBlitterAllocator* alloc) { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 90 | if (!supported(dst.info())) { |
| 91 | return nullptr; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 92 | } |
| 93 | if (paint.getShader()) { |
| 94 | return nullptr; // TODO: need to work out how shaders and their contexts work |
| 95 | } |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 96 | SkBlendMode blend = paint.getBlendMode(); |
| 97 | if (!SkBlendMode_AppendStages(blend)) { |
| 98 | return nullptr; // TODO |
| 99 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 100 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 101 | SkRasterPipeline shader, colorFilter; |
| 102 | if (paint.getColorFilter() && !paint.getColorFilter()->appendStages(&colorFilter)) { |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 103 | return nullptr; |
| 104 | } |
| 105 | |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 106 | uint32_t paintColor = paint.getColor(); |
| 107 | |
| 108 | SkColor4f color; |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 109 | if (SkImageInfoIsGammaCorrect(dst.info())) { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 110 | color = SkColor4f::FromColor(paintColor); |
| 111 | } else { |
| 112 | swizzle_rb(SkNx_cast<float>(Sk4b::Load(&paintColor)) * (1/255.0f)).store(&color); |
| 113 | } |
| 114 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 115 | auto blitter = alloc->createT<SkRasterPipelineBlitter>(dst, shader, blend, color.premul()); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 116 | |
| 117 | if (!paint.getShader()) { |
Mike Klein | fa9f241 | 2016-09-29 13:40:01 -0400 | [diff] [blame] | 118 | blitter->fShader.append(SkRasterPipeline::constant_color, &blitter->fPaintColor); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 119 | } |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 120 | blitter->fShader.extend(colorFilter); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 121 | |
| 122 | return blitter; |
| 123 | } |
| 124 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 125 | void SkRasterPipelineBlitter::append_load_d(SkRasterPipeline* p) const { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 126 | SkASSERT(supported(fDst.info())); |
| 127 | |
| 128 | switch (fDst.info().colorType()) { |
| 129 | case kN32_SkColorType: |
| 130 | if (fDst.info().gammaCloseToSRGB()) { |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 131 | p->append(SkRasterPipeline::load_d_srgb, &fDstPtr); |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 132 | } |
| 133 | break; |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 134 | case kRGBA_F16_SkColorType: |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 135 | p->append(SkRasterPipeline::load_d_f16, &fDstPtr); |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 136 | break; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 137 | case kRGB_565_SkColorType: |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 138 | p->append(SkRasterPipeline::load_d_565, &fDstPtr); |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 139 | break; |
| 140 | default: break; |
| 141 | } |
| 142 | } |
| 143 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 144 | void SkRasterPipelineBlitter::append_store(SkRasterPipeline* p) const { |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 145 | SkASSERT(supported(fDst.info())); |
| 146 | |
| 147 | switch (fDst.info().colorType()) { |
| 148 | case kN32_SkColorType: |
| 149 | if (fDst.info().gammaCloseToSRGB()) { |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 150 | p->append(SkRasterPipeline::store_srgb, &fDstPtr); |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 151 | } |
| 152 | break; |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 153 | case kRGBA_F16_SkColorType: |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 154 | p->append(SkRasterPipeline::store_f16, &fDstPtr); |
mtklein | 4e90e3e | 2016-07-25 14:35:31 -0700 | [diff] [blame] | 155 | break; |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 156 | case kRGB_565_SkColorType: |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 157 | p->append(SkRasterPipeline::store_565, &fDstPtr); |
mtklein | 8e4373f | 2016-07-22 14:20:27 -0700 | [diff] [blame] | 158 | break; |
| 159 | default: break; |
| 160 | } |
| 161 | } |
| 162 | |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 163 | void SkRasterPipelineBlitter::append_blend(SkRasterPipeline* p) const { |
| 164 | SkAssertResult(SkBlendMode_AppendStages(fBlend, p)); |
| 165 | } |
Mike Klein | e9f74b8 | 2016-10-25 13:31:21 -0400 | [diff] [blame] | 166 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 167 | void SkRasterPipelineBlitter::blitH(int x, int y, int w) { |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 168 | if (!fBlitH) { |
| 169 | SkRasterPipeline p; |
| 170 | p.extend(fShader); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 171 | this->append_load_d(&p); |
| 172 | this->append_blend(&p); |
| 173 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 174 | fBlitH = p.compile(); |
| 175 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 176 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 177 | fDstPtr = fDst.writable_addr(0,y); |
| 178 | fBlitH(x,w); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 179 | } |
| 180 | |
| 181 | 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] | 182 | if (!fBlitAntiH) { |
| 183 | SkRasterPipeline p; |
| 184 | p.extend(fShader); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 185 | this->append_load_d(&p); |
| 186 | this->append_blend(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 187 | p.append(SkRasterPipeline::lerp_constant_float, &fConstantCoverage); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 188 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 189 | fBlitAntiH = p.compile(); |
| 190 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 191 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 192 | fDstPtr = fDst.writable_addr(0,y); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 193 | for (int16_t run = *runs; run > 0; run = *runs) { |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 194 | fConstantCoverage = *aa * (1/255.0f); |
| 195 | fBlitAntiH(x, run); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 196 | |
| 197 | x += run; |
| 198 | runs += run; |
| 199 | aa += run; |
| 200 | } |
| 201 | } |
| 202 | |
| 203 | void SkRasterPipelineBlitter::blitMask(const SkMask& mask, const SkIRect& clip) { |
| 204 | if (mask.fFormat == SkMask::kBW_Format) { |
| 205 | // TODO: native BW masks? |
| 206 | return INHERITED::blitMask(mask, clip); |
| 207 | } |
| 208 | |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 209 | if (mask.fFormat == SkMask::kA8_Format && !fBlitMaskA8) { |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 210 | SkRasterPipeline p; |
| 211 | p.extend(fShader); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 212 | this->append_load_d(&p); |
| 213 | this->append_blend(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 214 | p.append(SkRasterPipeline::lerp_u8, &fMaskPtr); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 215 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 216 | fBlitMaskA8 = p.compile(); |
| 217 | } |
| 218 | |
| 219 | if (mask.fFormat == SkMask::kLCD16_Format && !fBlitMaskLCD16) { |
| 220 | SkRasterPipeline p; |
| 221 | p.extend(fShader); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 222 | this->append_load_d(&p); |
| 223 | this->append_blend(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 224 | p.append(SkRasterPipeline::lerp_565, &fMaskPtr); |
Mike Klein | e902f8d | 2016-10-26 15:32:26 -0400 | [diff] [blame^] | 225 | this->append_store(&p); |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 226 | fBlitMaskLCD16 = p.compile(); |
| 227 | } |
| 228 | |
| 229 | int x = clip.left(); |
| 230 | for (int y = clip.top(); y < clip.bottom(); y++) { |
| 231 | fDstPtr = fDst.writable_addr(0,y); |
| 232 | |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 233 | switch (mask.fFormat) { |
| 234 | case SkMask::kA8_Format: |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 235 | fMaskPtr = mask.getAddr8(x,y)-x; |
| 236 | fBlitMaskA8(x, clip.width()); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 237 | break; |
| 238 | case SkMask::kLCD16_Format: |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 239 | fMaskPtr = mask.getAddrLCD16(x,y)-x; |
| 240 | fBlitMaskLCD16(x, clip.width()); |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 241 | break; |
Mike Klein | bd3fe47 | 2016-10-25 15:43:46 -0400 | [diff] [blame] | 242 | default: |
| 243 | // TODO |
| 244 | break; |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 245 | } |
mtklein | 9a5c47f | 2016-07-22 11:05:04 -0700 | [diff] [blame] | 246 | } |
| 247 | } |