| /* |
| * Copyright 2017 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "include/core/SkBlendMode.h" |
| |
| #include "include/core/SkColor.h" |
| #include "include/core/SkPaint.h" |
| #include "src/base/SkVx.h" |
| #include "src/core/SkBlendModePriv.h" |
| #include "src/core/SkColorData.h" |
| #include "src/core/SkRasterPipeline.h" |
| #include "src/core/SkRasterPipelineOpContexts.h" |
| #include "src/core/SkRasterPipelineOpList.h" |
| |
| #include <optional> |
| |
| bool SkBlendMode_ShouldPreScaleCoverage(SkBlendMode mode, bool rgb_coverage) { |
| // The most important things we do here are: |
| // 1) never pre-scale with rgb coverage if the blend mode involves a source-alpha term; |
| // 2) always pre-scale Plus. |
| // |
| // When we pre-scale with rgb coverage, we scale each of source r,g,b, with a distinct value, |
| // and source alpha with one of those three values. This process destructively updates the |
| // source-alpha term, so we can't evaluate blend modes that need its original value. |
| // |
| // Plus always requires pre-scaling as a specific quirk of its implementation in |
| // SkRasterPipeline. This lets us put the clamp inside the blend mode itself rather |
| // than as a separate stage that'd come after the lerp. |
| // |
| // This function is a finer-grained breakdown of SkBlendMode_SupportsCoverageAsAlpha(). |
| switch (mode) { |
| case SkBlendMode::kDst: // d --> no sa term, ok! |
| case SkBlendMode::kDstOver: // d + s*inv(da) --> no sa term, ok! |
| case SkBlendMode::kPlus: // clamp(s+d) --> no sa term, ok! |
| return true; |
| |
| case SkBlendMode::kDstOut: // d * inv(sa) |
| case SkBlendMode::kSrcATop: // s*da + d*inv(sa) |
| case SkBlendMode::kSrcOver: // s + d*inv(sa) |
| case SkBlendMode::kXor: // s*inv(da) + d*inv(sa) |
| return !rgb_coverage; |
| |
| default: break; |
| } |
| return false; |
| } |
| |
| // Users of this function may want to switch to the rgb-coverage aware version above. |
| bool SkBlendMode_SupportsCoverageAsAlpha(SkBlendMode mode) { |
| return SkBlendMode_ShouldPreScaleCoverage(mode, false); |
| } |
| |
| bool SkBlendMode_AsCoeff(SkBlendMode mode, SkBlendModeCoeff* src, SkBlendModeCoeff* dst) { |
| struct CoeffRec { |
| SkBlendModeCoeff fSrc; |
| SkBlendModeCoeff fDst; |
| }; |
| |
| static constexpr CoeffRec kCoeffs[] = { |
| // For Porter-Duff blend functions, color = src * src coeff + dst * dst coeff |
| // src coeff dst coeff blend func |
| // ---------------------- ----------------------- ---------- |
| { SkBlendModeCoeff::kZero, SkBlendModeCoeff::kZero }, // clear |
| { SkBlendModeCoeff::kOne, SkBlendModeCoeff::kZero }, // src |
| { SkBlendModeCoeff::kZero, SkBlendModeCoeff::kOne }, // dst |
| { SkBlendModeCoeff::kOne, SkBlendModeCoeff::kISA }, // src-over |
| { SkBlendModeCoeff::kIDA, SkBlendModeCoeff::kOne }, // dst-over |
| { SkBlendModeCoeff::kDA, SkBlendModeCoeff::kZero }, // src-in |
| { SkBlendModeCoeff::kZero, SkBlendModeCoeff::kSA }, // dst-in |
| { SkBlendModeCoeff::kIDA, SkBlendModeCoeff::kZero }, // src-out |
| { SkBlendModeCoeff::kZero, SkBlendModeCoeff::kISA }, // dst-out |
| { SkBlendModeCoeff::kDA, SkBlendModeCoeff::kISA }, // src-atop |
| { SkBlendModeCoeff::kIDA, SkBlendModeCoeff::kSA }, // dst-atop |
| { SkBlendModeCoeff::kIDA, SkBlendModeCoeff::kISA }, // xor |
| |
| { SkBlendModeCoeff::kOne, SkBlendModeCoeff::kOne }, // plus |
| { SkBlendModeCoeff::kZero, SkBlendModeCoeff::kSC }, // modulate |
| { SkBlendModeCoeff::kOne, SkBlendModeCoeff::kISC }, // screen |
| }; |
| |
| if (mode > SkBlendMode::kScreen) { |
| return false; |
| } |
| if (src) { |
| *src = kCoeffs[static_cast<int>(mode)].fSrc; |
| } |
| if (dst) { |
| *dst = kCoeffs[static_cast<int>(mode)].fDst; |
| } |
| return true; |
| } |
| |
| void SkBlendMode_AppendStages(SkBlendMode mode, SkRasterPipeline* p) { |
| switch (mode) { |
| case SkBlendMode::kClear: p->append(SkRasterPipelineOp::clear); return; |
| case SkBlendMode::kSrc: return; // This stage is a no-op. |
| case SkBlendMode::kDst: p->append(SkRasterPipelineOp::move_dst_src); return; |
| case SkBlendMode::kSrcOver: p->append(SkRasterPipelineOp::srcover); return; |
| case SkBlendMode::kDstOver: p->append(SkRasterPipelineOp::dstover); return; |
| case SkBlendMode::kSrcIn: p->append(SkRasterPipelineOp::srcin); return; |
| case SkBlendMode::kDstIn: p->append(SkRasterPipelineOp::dstin); return; |
| case SkBlendMode::kSrcOut: p->append(SkRasterPipelineOp::srcout); return; |
| case SkBlendMode::kDstOut: p->append(SkRasterPipelineOp::dstout); return; |
| case SkBlendMode::kSrcATop: p->append(SkRasterPipelineOp::srcatop); return; |
| case SkBlendMode::kDstATop: p->append(SkRasterPipelineOp::dstatop); return; |
| case SkBlendMode::kXor: p->append(SkRasterPipelineOp::xor_); return; |
| case SkBlendMode::kPlus: p->append(SkRasterPipelineOp::plus_); return; |
| case SkBlendMode::kModulate: p->append(SkRasterPipelineOp::modulate); return; |
| |
| case SkBlendMode::kScreen: p->append(SkRasterPipelineOp::screen); return; |
| case SkBlendMode::kOverlay: p->append(SkRasterPipelineOp::overlay); return; |
| case SkBlendMode::kDarken: p->append(SkRasterPipelineOp::darken); return; |
| case SkBlendMode::kLighten: p->append(SkRasterPipelineOp::lighten); return; |
| case SkBlendMode::kColorDodge: p->append(SkRasterPipelineOp::colordodge); return; |
| case SkBlendMode::kColorBurn: p->append(SkRasterPipelineOp::colorburn); return; |
| case SkBlendMode::kHardLight: p->append(SkRasterPipelineOp::hardlight); return; |
| case SkBlendMode::kSoftLight: p->append(SkRasterPipelineOp::softlight); return; |
| case SkBlendMode::kDifference: p->append(SkRasterPipelineOp::difference); return; |
| case SkBlendMode::kExclusion: p->append(SkRasterPipelineOp::exclusion); return; |
| case SkBlendMode::kMultiply: p->append(SkRasterPipelineOp::multiply); return; |
| |
| case SkBlendMode::kHue: p->append(SkRasterPipelineOp::hue); return; |
| case SkBlendMode::kSaturation: p->append(SkRasterPipelineOp::saturation); return; |
| case SkBlendMode::kColor: p->append(SkRasterPipelineOp::color); return; |
| case SkBlendMode::kLuminosity: p->append(SkRasterPipelineOp::luminosity); return; |
| default: p->append(SkRasterPipelineOp::srcover); return; |
| } |
| SkUNREACHABLE; |
| } |
| |
| SkPMColor4f SkBlendMode_Apply(SkBlendMode mode, const SkPMColor4f& src, const SkPMColor4f& dst) { |
| // special-case simple/common modes... |
| switch (mode) { |
| case SkBlendMode::kClear: return SK_PMColor4fTRANSPARENT; |
| case SkBlendMode::kSrc: return src; |
| case SkBlendMode::kDst: return dst; |
| case SkBlendMode::kSrcOver: { |
| SkPMColor4f r; |
| (skvx::float4::Load(src.vec()) + skvx::float4::Load(dst.vec()) * (1-src.fA)).store(&r); |
| return r; |
| } |
| default: |
| break; |
| } |
| |
| SkRasterPipeline_<256> p; |
| SkPMColor4f src_storage = src, |
| dst_storage = dst, |
| res_storage; |
| SkRasterPipelineContexts::MemoryCtx src_ctx = {&src_storage, 0}, dst_ctx = {&dst_storage, 0}, |
| res_ctx = {&res_storage, 0}; |
| |
| p.append(SkRasterPipelineOp::load_f32, &dst_ctx); |
| p.append(SkRasterPipelineOp::move_src_dst); |
| p.append(SkRasterPipelineOp::load_f32, &src_ctx); |
| SkBlendMode_AppendStages(mode, &p); |
| p.append(SkRasterPipelineOp::store_f32, &res_ctx); |
| p.run(0,0, 1,1); |
| return res_storage; |
| } |
| |
| const char* SkBlendMode_Name(SkBlendMode bm) { |
| switch (bm) { |
| case SkBlendMode::kClear: return "Clear"; |
| case SkBlendMode::kSrc: return "Src"; |
| case SkBlendMode::kDst: return "Dst"; |
| case SkBlendMode::kSrcOver: return "SrcOver"; |
| case SkBlendMode::kDstOver: return "DstOver"; |
| case SkBlendMode::kSrcIn: return "SrcIn"; |
| case SkBlendMode::kDstIn: return "DstIn"; |
| case SkBlendMode::kSrcOut: return "SrcOut"; |
| case SkBlendMode::kDstOut: return "DstOut"; |
| case SkBlendMode::kSrcATop: return "SrcATop"; |
| case SkBlendMode::kDstATop: return "DstATop"; |
| case SkBlendMode::kXor: return "Xor"; |
| case SkBlendMode::kPlus: return "Plus"; |
| case SkBlendMode::kModulate: return "Modulate"; |
| case SkBlendMode::kScreen: return "Screen"; |
| |
| case SkBlendMode::kOverlay: return "Overlay"; |
| case SkBlendMode::kDarken: return "Darken"; |
| case SkBlendMode::kLighten: return "Lighten"; |
| case SkBlendMode::kColorDodge: return "ColorDodge"; |
| case SkBlendMode::kColorBurn: return "ColorBurn"; |
| case SkBlendMode::kHardLight: return "HardLight"; |
| case SkBlendMode::kSoftLight: return "SoftLight"; |
| case SkBlendMode::kDifference: return "Difference"; |
| case SkBlendMode::kExclusion: return "Exclusion"; |
| case SkBlendMode::kMultiply: return "Multiply"; |
| |
| case SkBlendMode::kHue: return "Hue"; |
| case SkBlendMode::kSaturation: return "Saturation"; |
| case SkBlendMode::kColor: return "Color"; |
| case SkBlendMode::kLuminosity: return "Luminosity"; |
| } |
| SkUNREACHABLE; |
| } |
| |
| static bool just_solid_color(const SkPaint& p) { |
| return SK_AlphaOPAQUE == p.getAlpha() && !p.getColorFilter() && !p.getShader(); |
| } |
| |
| SkBlendFastPath CheckFastPath(const SkPaint& paint, bool dstIsOpaque) { |
| const auto bm = paint.asBlendMode(); |
| if (!bm) { |
| return SkBlendFastPath::kNormal; |
| } |
| switch (bm.value()) { |
| case SkBlendMode::kSrcOver: |
| return SkBlendFastPath::kSrcOver; |
| case SkBlendMode::kSrc: |
| if (just_solid_color(paint)) { |
| return SkBlendFastPath::kSrcOver; |
| } |
| return SkBlendFastPath::kNormal; |
| case SkBlendMode::kDst: |
| return SkBlendFastPath::kSkipDrawing; |
| case SkBlendMode::kDstOver: |
| if (dstIsOpaque) { |
| return SkBlendFastPath::kSkipDrawing; |
| } |
| return SkBlendFastPath::kNormal; |
| case SkBlendMode::kSrcIn: |
| if (dstIsOpaque && just_solid_color(paint)) { |
| return SkBlendFastPath::kSrcOver; |
| } |
| return SkBlendFastPath::kNormal; |
| case SkBlendMode::kDstIn: |
| if (just_solid_color(paint)) { |
| return SkBlendFastPath::kSkipDrawing; |
| } |
| return SkBlendFastPath::kNormal; |
| default: |
| return SkBlendFastPath::kNormal; |
| } |
| } |