| /* |
| * Copyright 2016 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "src/core/SkRasterPipeline.h" |
| |
| #include "include/core/SkColorType.h" |
| #include "include/core/SkImageInfo.h" |
| #include "include/core/SkMatrix.h" |
| #include "include/private/base/SkTemplates.h" |
| #include "modules/skcms/skcms.h" |
| #include "src/base/SkVx.h" |
| #include "src/core/SkImageInfoPriv.h" |
| #include "src/core/SkOpts.h" |
| |
| #include <algorithm> |
| #include <cstring> |
| #include <vector> |
| |
| using namespace skia_private; |
| using Op = SkRasterPipelineOp; |
| |
| bool gForceHighPrecisionRasterPipeline; |
| |
| SkRasterPipeline::SkRasterPipeline(SkArenaAlloc* alloc) : fAlloc(alloc) { |
| this->reset(); |
| } |
| void SkRasterPipeline::reset() { |
| fRewindCtx = nullptr; |
| fStages = nullptr; |
| fNumStages = 0; |
| } |
| |
| void SkRasterPipeline::append(SkRasterPipelineOp op, void* ctx) { |
| SkASSERT(op != Op::uniform_color); // Please use append_constant_color(). |
| SkASSERT(op != Op::unbounded_uniform_color); // Please use append_constant_color(). |
| SkASSERT(op != Op::set_rgb); // Please use append_set_rgb(). |
| SkASSERT(op != Op::unbounded_set_rgb); // Please use append_set_rgb(). |
| SkASSERT(op != Op::parametric); // Please use append_transfer_function(). |
| SkASSERT(op != Op::gamma_); // Please use append_transfer_function(). |
| SkASSERT(op != Op::PQish); // Please use append_transfer_function(). |
| SkASSERT(op != Op::HLGish); // Please use append_transfer_function(). |
| SkASSERT(op != Op::HLGinvish); // Please use append_transfer_function(). |
| SkASSERT(op != Op::stack_checkpoint); // Please use append_stack_rewind(). |
| SkASSERT(op != Op::stack_rewind); // Please use append_stack_rewind(). |
| this->unchecked_append(op, ctx); |
| } |
| void SkRasterPipeline::unchecked_append(SkRasterPipelineOp op, void* ctx) { |
| fStages = fAlloc->make<StageList>(StageList{fStages, op, ctx}); |
| fNumStages += 1; |
| } |
| void SkRasterPipeline::append(SkRasterPipelineOp op, uintptr_t ctx) { |
| void* ptrCtx; |
| memcpy(&ptrCtx, &ctx, sizeof(ctx)); |
| this->append(op, ptrCtx); |
| } |
| |
| void SkRasterPipeline::extend(const SkRasterPipeline& src) { |
| if (src.empty()) { |
| return; |
| } |
| // Create a rewind context if `src` has one already, but we don't. If we _do_ already have one, |
| // we need to keep it, since we already have rewind ops that reference it. Either way, we need |
| // to rewrite all the rewind ops to point to _our_ rewind context; we only get that checkpoint. |
| if (src.fRewindCtx && !fRewindCtx) { |
| fRewindCtx = fAlloc->make<SkRasterPipeline_RewindCtx>(); |
| } |
| auto stages = fAlloc->makeArrayDefault<StageList>(src.fNumStages); |
| |
| int n = src.fNumStages; |
| const StageList* st = src.fStages; |
| while (n --> 1) { |
| stages[n] = *st; |
| stages[n].prev = &stages[n-1]; |
| |
| if (stages[n].stage == Op::stack_rewind) { |
| // We make sure that all stack rewinds use _our_ stack context. |
| stages[n].ctx = fRewindCtx; |
| } |
| |
| st = st->prev; |
| } |
| stages[0] = *st; |
| stages[0].prev = fStages; |
| |
| fStages = &stages[src.fNumStages - 1]; |
| fNumStages += src.fNumStages; |
| } |
| |
| const char* SkRasterPipeline::GetOpName(SkRasterPipelineOp op) { |
| const char* name = ""; |
| switch (op) { |
| #define M(x) case Op::x: name = #x; break; |
| SK_RASTER_PIPELINE_OPS_ALL(M) |
| #undef M |
| } |
| return name; |
| } |
| |
| void SkRasterPipeline::dump() const { |
| SkDebugf("SkRasterPipeline, %d stages\n", fNumStages); |
| std::vector<const char*> stages; |
| for (auto st = fStages; st; st = st->prev) { |
| stages.push_back(GetOpName(st->stage)); |
| } |
| std::reverse(stages.begin(), stages.end()); |
| for (const char* name : stages) { |
| SkDebugf("\t%s\n", name); |
| } |
| SkDebugf("\n"); |
| } |
| |
| void SkRasterPipeline::append_set_rgb(SkArenaAlloc* alloc, const float rgb[3]) { |
| auto arg = alloc->makeArrayDefault<float>(3); |
| arg[0] = rgb[0]; |
| arg[1] = rgb[1]; |
| arg[2] = rgb[2]; |
| |
| auto op = Op::unbounded_set_rgb; |
| if (0 <= rgb[0] && rgb[0] <= 1 && |
| 0 <= rgb[1] && rgb[1] <= 1 && |
| 0 <= rgb[2] && rgb[2] <= 1) |
| { |
| op = Op::set_rgb; |
| } |
| |
| this->unchecked_append(op, arg); |
| } |
| |
| void SkRasterPipeline::append_constant_color(SkArenaAlloc* alloc, const float rgba[4]) { |
| // r,g,b might be outside [0,1], but alpha should probably always be in [0,1]. |
| SkASSERT(0 <= rgba[3] && rgba[3] <= 1); |
| |
| if (rgba[0] == 0 && rgba[1] == 0 && rgba[2] == 0 && rgba[3] == 1) { |
| this->append(Op::black_color); |
| } else if (rgba[0] == 1 && rgba[1] == 1 && rgba[2] == 1 && rgba[3] == 1) { |
| this->append(Op::white_color); |
| } else { |
| auto ctx = alloc->make<SkRasterPipeline_UniformColorCtx>(); |
| skvx::float4 color = skvx::float4::Load(rgba); |
| color.store(&ctx->r); |
| |
| // uniform_color requires colors in range and can go lowp, |
| // while unbounded_uniform_color supports out-of-range colors too but not lowp. |
| if (0 <= rgba[0] && rgba[0] <= rgba[3] && |
| 0 <= rgba[1] && rgba[1] <= rgba[3] && |
| 0 <= rgba[2] && rgba[2] <= rgba[3]) { |
| // To make loads more direct, we store 8-bit values in 16-bit slots. |
| color = color * 255.0f + 0.5f; |
| ctx->rgba[0] = (uint16_t)color[0]; |
| ctx->rgba[1] = (uint16_t)color[1]; |
| ctx->rgba[2] = (uint16_t)color[2]; |
| ctx->rgba[3] = (uint16_t)color[3]; |
| this->unchecked_append(Op::uniform_color, ctx); |
| } else { |
| this->unchecked_append(Op::unbounded_uniform_color, ctx); |
| } |
| } |
| } |
| |
| void SkRasterPipeline::append_matrix(SkArenaAlloc* alloc, const SkMatrix& matrix) { |
| SkMatrix::TypeMask mt = matrix.getType(); |
| |
| if (mt == SkMatrix::kIdentity_Mask) { |
| return; |
| } |
| if (mt == SkMatrix::kTranslate_Mask) { |
| float* trans = alloc->makeArrayDefault<float>(2); |
| trans[0] = matrix.getTranslateX(); |
| trans[1] = matrix.getTranslateY(); |
| this->append(Op::matrix_translate, trans); |
| } else if ((mt | (SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask)) == |
| (SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask)) { |
| float* scaleTrans = alloc->makeArrayDefault<float>(4); |
| scaleTrans[0] = matrix.getScaleX(); |
| scaleTrans[1] = matrix.getScaleY(); |
| scaleTrans[2] = matrix.getTranslateX(); |
| scaleTrans[3] = matrix.getTranslateY(); |
| this->append(Op::matrix_scale_translate, scaleTrans); |
| } else { |
| float* storage = alloc->makeArrayDefault<float>(9); |
| matrix.get9(storage); |
| if (!matrix.hasPerspective()) { |
| // note: asAffine and the 2x3 stage really only need 6 entries |
| this->append(Op::matrix_2x3, storage); |
| } else { |
| this->append(Op::matrix_perspective, storage); |
| } |
| } |
| } |
| |
| void SkRasterPipeline::append_load(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { |
| switch (ct) { |
| case kUnknown_SkColorType: SkASSERT(false); break; |
| |
| case kAlpha_8_SkColorType: this->append(Op::load_a8, ctx); break; |
| case kA16_unorm_SkColorType: this->append(Op::load_a16, ctx); break; |
| case kA16_float_SkColorType: this->append(Op::load_af16, ctx); break; |
| case kRGB_565_SkColorType: this->append(Op::load_565, ctx); break; |
| case kARGB_4444_SkColorType: this->append(Op::load_4444, ctx); break; |
| case kR8G8_unorm_SkColorType: this->append(Op::load_rg88, ctx); break; |
| case kR16G16_unorm_SkColorType: this->append(Op::load_rg1616, ctx); break; |
| case kR16G16_float_SkColorType: this->append(Op::load_rgf16, ctx); break; |
| case kRGBA_8888_SkColorType: this->append(Op::load_8888, ctx); break; |
| case kRGBA_1010102_SkColorType: this->append(Op::load_1010102, ctx); break; |
| case kR16G16B16A16_unorm_SkColorType:this->append(Op::load_16161616,ctx); break; |
| case kRGBA_F16Norm_SkColorType: |
| case kRGBA_F16_SkColorType: this->append(Op::load_f16, ctx); break; |
| case kRGBA_F32_SkColorType: this->append(Op::load_f32, ctx); break; |
| |
| case kGray_8_SkColorType: this->append(Op::load_a8, ctx); |
| this->append(Op::alpha_to_gray); |
| break; |
| |
| case kR8_unorm_SkColorType: this->append(Op::load_a8, ctx); |
| this->append(Op::alpha_to_red); |
| break; |
| |
| case kRGB_888x_SkColorType: this->append(Op::load_8888, ctx); |
| this->append(Op::force_opaque); |
| break; |
| |
| case kBGRA_1010102_SkColorType: this->append(Op::load_1010102, ctx); |
| this->append(Op::swap_rb); |
| break; |
| |
| case kRGB_101010x_SkColorType: this->append(Op::load_1010102, ctx); |
| this->append(Op::force_opaque); |
| break; |
| |
| case kBGR_101010x_SkColorType: this->append(Op::load_1010102, ctx); |
| this->append(Op::force_opaque); |
| this->append(Op::swap_rb); |
| break; |
| |
| case kBGR_101010x_XR_SkColorType: this->append(Op::load_1010102_xr, ctx); |
| this->append(Op::force_opaque); |
| this->append(Op::swap_rb); |
| break; |
| |
| case kBGRA_8888_SkColorType: this->append(Op::load_8888, ctx); |
| this->append(Op::swap_rb); |
| break; |
| |
| case kSRGBA_8888_SkColorType: |
| this->append(Op::load_8888, ctx); |
| this->append_transfer_function(*skcms_sRGB_TransferFunction()); |
| break; |
| } |
| } |
| |
| void SkRasterPipeline::append_load_dst(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { |
| switch (ct) { |
| case kUnknown_SkColorType: SkASSERT(false); break; |
| |
| case kAlpha_8_SkColorType: this->append(Op::load_a8_dst, ctx); break; |
| case kA16_unorm_SkColorType: this->append(Op::load_a16_dst, ctx); break; |
| case kA16_float_SkColorType: this->append(Op::load_af16_dst, ctx); break; |
| case kRGB_565_SkColorType: this->append(Op::load_565_dst, ctx); break; |
| case kARGB_4444_SkColorType: this->append(Op::load_4444_dst, ctx); break; |
| case kR8G8_unorm_SkColorType: this->append(Op::load_rg88_dst, ctx); break; |
| case kR16G16_unorm_SkColorType: this->append(Op::load_rg1616_dst, ctx); break; |
| case kR16G16_float_SkColorType: this->append(Op::load_rgf16_dst, ctx); break; |
| case kRGBA_8888_SkColorType: this->append(Op::load_8888_dst, ctx); break; |
| case kRGBA_1010102_SkColorType: this->append(Op::load_1010102_dst, ctx); break; |
| case kR16G16B16A16_unorm_SkColorType: this->append(Op::load_16161616_dst,ctx); break; |
| case kRGBA_F16Norm_SkColorType: |
| case kRGBA_F16_SkColorType: this->append(Op::load_f16_dst, ctx); break; |
| case kRGBA_F32_SkColorType: this->append(Op::load_f32_dst, ctx); break; |
| |
| case kGray_8_SkColorType: this->append(Op::load_a8_dst, ctx); |
| this->append(Op::alpha_to_gray_dst); |
| break; |
| |
| case kR8_unorm_SkColorType: this->append(Op::load_a8_dst, ctx); |
| this->append(Op::alpha_to_red_dst); |
| break; |
| |
| case kRGB_888x_SkColorType: this->append(Op::load_8888_dst, ctx); |
| this->append(Op::force_opaque_dst); |
| break; |
| |
| case kBGRA_1010102_SkColorType: this->append(Op::load_1010102_dst, ctx); |
| this->append(Op::swap_rb_dst); |
| break; |
| |
| case kRGB_101010x_SkColorType: this->append(Op::load_1010102_dst, ctx); |
| this->append(Op::force_opaque_dst); |
| break; |
| |
| case kBGR_101010x_SkColorType: this->append(Op::load_1010102_dst, ctx); |
| this->append(Op::force_opaque_dst); |
| this->append(Op::swap_rb_dst); |
| break; |
| |
| case kBGR_101010x_XR_SkColorType: this->append(Op::load_1010102_xr_dst, ctx); |
| this->append(Op::force_opaque_dst); |
| this->append(Op::swap_rb_dst); |
| break; |
| |
| case kBGRA_8888_SkColorType: this->append(Op::load_8888_dst, ctx); |
| this->append(Op::swap_rb_dst); |
| break; |
| |
| case kSRGBA_8888_SkColorType: |
| // TODO: We could remove the double-swap if we had _dst versions of all the TF stages |
| this->append(Op::load_8888_dst, ctx); |
| this->append(Op::swap_src_dst); |
| this->append_transfer_function(*skcms_sRGB_TransferFunction()); |
| this->append(Op::swap_src_dst); |
| break; |
| } |
| } |
| |
| void SkRasterPipeline::append_store(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { |
| switch (ct) { |
| case kUnknown_SkColorType: SkASSERT(false); break; |
| |
| case kAlpha_8_SkColorType: this->append(Op::store_a8, ctx); break; |
| case kR8_unorm_SkColorType: this->append(Op::store_r8, ctx); break; |
| case kA16_unorm_SkColorType: this->append(Op::store_a16, ctx); break; |
| case kA16_float_SkColorType: this->append(Op::store_af16, ctx); break; |
| case kRGB_565_SkColorType: this->append(Op::store_565, ctx); break; |
| case kARGB_4444_SkColorType: this->append(Op::store_4444, ctx); break; |
| case kR8G8_unorm_SkColorType: this->append(Op::store_rg88, ctx); break; |
| case kR16G16_unorm_SkColorType: this->append(Op::store_rg1616, ctx); break; |
| case kR16G16_float_SkColorType: this->append(Op::store_rgf16, ctx); break; |
| case kRGBA_8888_SkColorType: this->append(Op::store_8888, ctx); break; |
| case kRGBA_1010102_SkColorType: this->append(Op::store_1010102, ctx); break; |
| case kR16G16B16A16_unorm_SkColorType: this->append(Op::store_16161616,ctx); break; |
| case kRGBA_F16Norm_SkColorType: |
| case kRGBA_F16_SkColorType: this->append(Op::store_f16, ctx); break; |
| case kRGBA_F32_SkColorType: this->append(Op::store_f32, ctx); break; |
| |
| case kRGB_888x_SkColorType: this->append(Op::force_opaque); |
| this->append(Op::store_8888, ctx); |
| break; |
| |
| case kBGRA_1010102_SkColorType: this->append(Op::swap_rb); |
| this->append(Op::store_1010102, ctx); |
| break; |
| |
| case kRGB_101010x_SkColorType: this->append(Op::force_opaque); |
| this->append(Op::store_1010102, ctx); |
| break; |
| |
| case kBGR_101010x_SkColorType: this->append(Op::force_opaque); |
| this->append(Op::swap_rb); |
| this->append(Op::store_1010102, ctx); |
| break; |
| |
| case kBGR_101010x_XR_SkColorType: this->append(Op::force_opaque); |
| this->append(Op::swap_rb); |
| this->append(Op::store_1010102_xr, ctx); |
| break; |
| |
| case kGray_8_SkColorType: this->append(Op::bt709_luminance_or_luma_to_alpha); |
| this->append(Op::store_a8, ctx); |
| break; |
| |
| case kBGRA_8888_SkColorType: this->append(Op::swap_rb); |
| this->append(Op::store_8888, ctx); |
| break; |
| |
| case kSRGBA_8888_SkColorType: |
| this->append_transfer_function(*skcms_sRGB_Inverse_TransferFunction()); |
| this->append(Op::store_8888, ctx); |
| break; |
| } |
| } |
| |
| void SkRasterPipeline::append_transfer_function(const skcms_TransferFunction& tf) { |
| void* ctx = const_cast<void*>(static_cast<const void*>(&tf)); |
| switch (skcms_TransferFunction_getType(&tf)) { |
| case skcms_TFType_Invalid: SkASSERT(false); break; |
| |
| case skcms_TFType_sRGBish: |
| if (tf.a == 1 && tf.b == 0 && tf.c == 0 && tf.d == 0 && tf.e == 0 && tf.f == 0) { |
| this->unchecked_append(Op::gamma_, ctx); |
| } else { |
| this->unchecked_append(Op::parametric, ctx); |
| } |
| break; |
| case skcms_TFType_PQish: this->unchecked_append(Op::PQish, ctx); break; |
| case skcms_TFType_HLGish: this->unchecked_append(Op::HLGish, ctx); break; |
| case skcms_TFType_HLGinvish: this->unchecked_append(Op::HLGinvish, ctx); break; |
| } |
| } |
| |
| // GPUs clamp all color channels to the limits of the format just before the blend step. To match |
| // that auto-clamp, the RP blitter uses this helper immediately before appending blending stages. |
| void SkRasterPipeline::append_clamp_if_normalized(const SkImageInfo& info) { |
| if (SkColorTypeIsNormalized(info.colorType())) { |
| this->unchecked_append(Op::clamp_01, nullptr); |
| } |
| } |
| |
| void SkRasterPipeline::append_stack_rewind() { |
| if (!fRewindCtx) { |
| fRewindCtx = fAlloc->make<SkRasterPipeline_RewindCtx>(); |
| } |
| this->unchecked_append(Op::stack_rewind, fRewindCtx); |
| } |
| |
| static void prepend_to_pipeline(SkRasterPipelineStage*& ip, SkOpts::StageFn stageFn, void* ctx) { |
| --ip; |
| ip->fn = stageFn; |
| ip->ctx = ctx; |
| } |
| |
| bool SkRasterPipeline::build_lowp_pipeline(SkRasterPipelineStage* ip) const { |
| if (gForceHighPrecisionRasterPipeline || fRewindCtx) { |
| return false; |
| } |
| // Stages are stored backwards in fStages; to compensate, we assemble the pipeline in reverse |
| // here, back to front. |
| prepend_to_pipeline(ip, SkOpts::just_return_lowp, /*ctx=*/nullptr); |
| for (const StageList* st = fStages; st; st = st->prev) { |
| int opIndex = (int)st->stage; |
| if (opIndex >= kNumRasterPipelineLowpOps || !SkOpts::ops_lowp[opIndex]) { |
| // This program contains a stage that doesn't exist in lowp. |
| return false; |
| } |
| prepend_to_pipeline(ip, SkOpts::ops_lowp[opIndex], st->ctx); |
| } |
| return true; |
| } |
| |
| void SkRasterPipeline::build_highp_pipeline(SkRasterPipelineStage* ip) const { |
| // We assemble the pipeline in reverse, since the stage list is stored backwards. |
| prepend_to_pipeline(ip, SkOpts::just_return_highp, /*ctx=*/nullptr); |
| for (const StageList* st = fStages; st; st = st->prev) { |
| int opIndex = (int)st->stage; |
| prepend_to_pipeline(ip, SkOpts::ops_highp[opIndex], st->ctx); |
| } |
| |
| // stack_checkpoint and stack_rewind are only implemented in highp. We only need these stages |
| // when generating long (or looping) pipelines from SkSL. The other stages used by the SkSL |
| // Raster Pipeline generator will only have highp implementations, because we can't execute SkSL |
| // code without floating point. |
| if (fRewindCtx) { |
| const int rewindIndex = (int)Op::stack_checkpoint; |
| prepend_to_pipeline(ip, SkOpts::ops_highp[rewindIndex], fRewindCtx); |
| } |
| } |
| |
| SkRasterPipeline::StartPipelineFn SkRasterPipeline::build_pipeline( |
| SkRasterPipelineStage* ip) const { |
| // We try to build a lowp pipeline first; if that fails, we fall back to a highp float pipeline. |
| if (this->build_lowp_pipeline(ip)) { |
| return SkOpts::start_pipeline_lowp; |
| } |
| |
| this->build_highp_pipeline(ip); |
| return SkOpts::start_pipeline_highp; |
| } |
| |
| int SkRasterPipeline::stages_needed() const { |
| // Add 1 to budget for a `just_return` stage at the end. |
| int stages = fNumStages + 1; |
| |
| // If we have any stack_rewind stages, we will need to inject a stack_checkpoint stage. |
| if (fRewindCtx) { |
| stages += 1; |
| } |
| return stages; |
| } |
| |
| void SkRasterPipeline::run(size_t x, size_t y, size_t w, size_t h) const { |
| if (this->empty()) { |
| return; |
| } |
| |
| int stagesNeeded = this->stages_needed(); |
| |
| // Best to not use fAlloc here... we can't bound how often run() will be called. |
| AutoSTMalloc<32, SkRasterPipelineStage> program(stagesNeeded); |
| |
| auto start_pipeline = this->build_pipeline(program.get() + stagesNeeded); |
| start_pipeline(x,y,x+w,y+h, program.get()); |
| } |
| |
| std::function<void(size_t, size_t, size_t, size_t)> SkRasterPipeline::compile() const { |
| if (this->empty()) { |
| return [](size_t, size_t, size_t, size_t) {}; |
| } |
| |
| int stagesNeeded = this->stages_needed(); |
| |
| SkRasterPipelineStage* program = fAlloc->makeArray<SkRasterPipelineStage>(stagesNeeded); |
| |
| auto start_pipeline = this->build_pipeline(program + stagesNeeded); |
| return [=](size_t x, size_t y, size_t w, size_t h) { |
| start_pipeline(x,y,x+w,y+h, program); |
| }; |
| } |