1cb93a386Sopenharmony_ci/* 2cb93a386Sopenharmony_ci * Copyright 2016 Google Inc. 3cb93a386Sopenharmony_ci * 4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be 5cb93a386Sopenharmony_ci * found in the LICENSE file. 6cb93a386Sopenharmony_ci */ 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "include/private/SkImageInfoPriv.h" 9cb93a386Sopenharmony_ci#include "include/private/SkNx.h" 10cb93a386Sopenharmony_ci#include "include/private/SkTemplates.h" 11cb93a386Sopenharmony_ci#include "src/core/SkColorSpacePriv.h" 12cb93a386Sopenharmony_ci#include "src/core/SkOpts.h" 13cb93a386Sopenharmony_ci#include "src/core/SkRasterPipeline.h" 14cb93a386Sopenharmony_ci#include <algorithm> 15cb93a386Sopenharmony_ci 16cb93a386Sopenharmony_cibool gForceHighPrecisionRasterPipeline; 17cb93a386Sopenharmony_ci 18cb93a386Sopenharmony_ciSkRasterPipeline::SkRasterPipeline(SkArenaAlloc* alloc) : fAlloc(alloc) { 19cb93a386Sopenharmony_ci this->reset(); 20cb93a386Sopenharmony_ci} 21cb93a386Sopenharmony_civoid SkRasterPipeline::reset() { 22cb93a386Sopenharmony_ci fStages = nullptr; 23cb93a386Sopenharmony_ci fNumStages = 0; 24cb93a386Sopenharmony_ci fSlotsNeeded = 1; // We always need one extra slot for just_return(). 25cb93a386Sopenharmony_ci} 26cb93a386Sopenharmony_ci 27cb93a386Sopenharmony_civoid SkRasterPipeline::append(StockStage stage, void* ctx) { 28cb93a386Sopenharmony_ci SkASSERT(stage != uniform_color); // Please use append_constant_color(). 29cb93a386Sopenharmony_ci SkASSERT(stage != unbounded_uniform_color); // Please use append_constant_color(). 30cb93a386Sopenharmony_ci SkASSERT(stage != set_rgb); // Please use append_set_rgb(). 31cb93a386Sopenharmony_ci SkASSERT(stage != unbounded_set_rgb); // Please use append_set_rgb(). 32cb93a386Sopenharmony_ci SkASSERT(stage != clamp_gamut); // Please use append_gamut_clamp_if_normalized(). 33cb93a386Sopenharmony_ci SkASSERT(stage != parametric); // Please use append_transfer_function(). 34cb93a386Sopenharmony_ci SkASSERT(stage != gamma_); // Please use append_transfer_function(). 35cb93a386Sopenharmony_ci SkASSERT(stage != PQish); // Please use append_transfer_function(). 36cb93a386Sopenharmony_ci SkASSERT(stage != HLGish); // Please use append_transfer_function(). 37cb93a386Sopenharmony_ci SkASSERT(stage != HLGinvish); // Please use append_transfer_function(). 38cb93a386Sopenharmony_ci this->unchecked_append(stage, ctx); 39cb93a386Sopenharmony_ci} 40cb93a386Sopenharmony_civoid SkRasterPipeline::unchecked_append(StockStage stage, void* ctx) { 41cb93a386Sopenharmony_ci fStages = fAlloc->make<StageList>( StageList{fStages, stage, ctx} ); 42cb93a386Sopenharmony_ci fNumStages += 1; 43cb93a386Sopenharmony_ci fSlotsNeeded += ctx ? 2 : 1; 44cb93a386Sopenharmony_ci} 45cb93a386Sopenharmony_civoid SkRasterPipeline::append(StockStage stage, uintptr_t ctx) { 46cb93a386Sopenharmony_ci void* ptrCtx; 47cb93a386Sopenharmony_ci memcpy(&ptrCtx, &ctx, sizeof(ctx)); 48cb93a386Sopenharmony_ci this->append(stage, ptrCtx); 49cb93a386Sopenharmony_ci} 50cb93a386Sopenharmony_ci 51cb93a386Sopenharmony_civoid SkRasterPipeline::extend(const SkRasterPipeline& src) { 52cb93a386Sopenharmony_ci if (src.empty()) { 53cb93a386Sopenharmony_ci return; 54cb93a386Sopenharmony_ci } 55cb93a386Sopenharmony_ci auto stages = fAlloc->makeArrayDefault<StageList>(src.fNumStages); 56cb93a386Sopenharmony_ci 57cb93a386Sopenharmony_ci int n = src.fNumStages; 58cb93a386Sopenharmony_ci const StageList* st = src.fStages; 59cb93a386Sopenharmony_ci while (n --> 1) { 60cb93a386Sopenharmony_ci stages[n] = *st; 61cb93a386Sopenharmony_ci stages[n].prev = &stages[n-1]; 62cb93a386Sopenharmony_ci st = st->prev; 63cb93a386Sopenharmony_ci } 64cb93a386Sopenharmony_ci stages[0] = *st; 65cb93a386Sopenharmony_ci stages[0].prev = fStages; 66cb93a386Sopenharmony_ci 67cb93a386Sopenharmony_ci fStages = &stages[src.fNumStages - 1]; 68cb93a386Sopenharmony_ci fNumStages += src.fNumStages; 69cb93a386Sopenharmony_ci fSlotsNeeded += src.fSlotsNeeded - 1; // Don't double count just_returns(). 70cb93a386Sopenharmony_ci} 71cb93a386Sopenharmony_ci 72cb93a386Sopenharmony_civoid SkRasterPipeline::dump() const { 73cb93a386Sopenharmony_ci SkDebugf("SkRasterPipeline, %d stages\n", fNumStages); 74cb93a386Sopenharmony_ci std::vector<const char*> stages; 75cb93a386Sopenharmony_ci for (auto st = fStages; st; st = st->prev) { 76cb93a386Sopenharmony_ci const char* name = ""; 77cb93a386Sopenharmony_ci switch (st->stage) { 78cb93a386Sopenharmony_ci #define M(x) case x: name = #x; break; 79cb93a386Sopenharmony_ci SK_RASTER_PIPELINE_STAGES(M) 80cb93a386Sopenharmony_ci #undef M 81cb93a386Sopenharmony_ci } 82cb93a386Sopenharmony_ci stages.push_back(name); 83cb93a386Sopenharmony_ci } 84cb93a386Sopenharmony_ci std::reverse(stages.begin(), stages.end()); 85cb93a386Sopenharmony_ci for (const char* name : stages) { 86cb93a386Sopenharmony_ci SkDebugf("\t%s\n", name); 87cb93a386Sopenharmony_ci } 88cb93a386Sopenharmony_ci SkDebugf("\n"); 89cb93a386Sopenharmony_ci} 90cb93a386Sopenharmony_ci 91cb93a386Sopenharmony_civoid SkRasterPipeline::append_set_rgb(SkArenaAlloc* alloc, const float rgb[3]) { 92cb93a386Sopenharmony_ci auto arg = alloc->makeArrayDefault<float>(3); 93cb93a386Sopenharmony_ci arg[0] = rgb[0]; 94cb93a386Sopenharmony_ci arg[1] = rgb[1]; 95cb93a386Sopenharmony_ci arg[2] = rgb[2]; 96cb93a386Sopenharmony_ci 97cb93a386Sopenharmony_ci auto stage = unbounded_set_rgb; 98cb93a386Sopenharmony_ci if (0 <= rgb[0] && rgb[0] <= 1 && 99cb93a386Sopenharmony_ci 0 <= rgb[1] && rgb[1] <= 1 && 100cb93a386Sopenharmony_ci 0 <= rgb[2] && rgb[2] <= 1) 101cb93a386Sopenharmony_ci { 102cb93a386Sopenharmony_ci stage = set_rgb; 103cb93a386Sopenharmony_ci } 104cb93a386Sopenharmony_ci 105cb93a386Sopenharmony_ci this->unchecked_append(stage, arg); 106cb93a386Sopenharmony_ci} 107cb93a386Sopenharmony_ci 108cb93a386Sopenharmony_civoid SkRasterPipeline::append_constant_color(SkArenaAlloc* alloc, const float rgba[4]) { 109cb93a386Sopenharmony_ci // r,g,b might be outside [0,1], but alpha should probably always be in [0,1]. 110cb93a386Sopenharmony_ci SkASSERT(0 <= rgba[3] && rgba[3] <= 1); 111cb93a386Sopenharmony_ci 112cb93a386Sopenharmony_ci if (rgba[0] == 0 && rgba[1] == 0 && rgba[2] == 0 && rgba[3] == 1) { 113cb93a386Sopenharmony_ci this->append(black_color); 114cb93a386Sopenharmony_ci } else if (rgba[0] == 1 && rgba[1] == 1 && rgba[2] == 1 && rgba[3] == 1) { 115cb93a386Sopenharmony_ci this->append(white_color); 116cb93a386Sopenharmony_ci } else { 117cb93a386Sopenharmony_ci auto ctx = alloc->make<SkRasterPipeline_UniformColorCtx>(); 118cb93a386Sopenharmony_ci Sk4f color = Sk4f::Load(rgba); 119cb93a386Sopenharmony_ci color.store(&ctx->r); 120cb93a386Sopenharmony_ci 121cb93a386Sopenharmony_ci // uniform_color requires colors in range and can go lowp, 122cb93a386Sopenharmony_ci // while unbounded_uniform_color supports out-of-range colors too but not lowp. 123cb93a386Sopenharmony_ci if (0 <= rgba[0] && rgba[0] <= rgba[3] && 124cb93a386Sopenharmony_ci 0 <= rgba[1] && rgba[1] <= rgba[3] && 125cb93a386Sopenharmony_ci 0 <= rgba[2] && rgba[2] <= rgba[3]) { 126cb93a386Sopenharmony_ci // To make loads more direct, we store 8-bit values in 16-bit slots. 127cb93a386Sopenharmony_ci color = color * 255.0f + 0.5f; 128cb93a386Sopenharmony_ci ctx->rgba[0] = (uint16_t)color[0]; 129cb93a386Sopenharmony_ci ctx->rgba[1] = (uint16_t)color[1]; 130cb93a386Sopenharmony_ci ctx->rgba[2] = (uint16_t)color[2]; 131cb93a386Sopenharmony_ci ctx->rgba[3] = (uint16_t)color[3]; 132cb93a386Sopenharmony_ci this->unchecked_append(uniform_color, ctx); 133cb93a386Sopenharmony_ci } else { 134cb93a386Sopenharmony_ci this->unchecked_append(unbounded_uniform_color, ctx); 135cb93a386Sopenharmony_ci } 136cb93a386Sopenharmony_ci } 137cb93a386Sopenharmony_ci} 138cb93a386Sopenharmony_ci 139cb93a386Sopenharmony_civoid SkRasterPipeline::append_matrix(SkArenaAlloc* alloc, const SkMatrix& matrix) { 140cb93a386Sopenharmony_ci SkMatrix::TypeMask mt = matrix.getType(); 141cb93a386Sopenharmony_ci 142cb93a386Sopenharmony_ci if (mt == SkMatrix::kIdentity_Mask) { 143cb93a386Sopenharmony_ci return; 144cb93a386Sopenharmony_ci } 145cb93a386Sopenharmony_ci if (mt == SkMatrix::kTranslate_Mask) { 146cb93a386Sopenharmony_ci float* trans = alloc->makeArrayDefault<float>(2); 147cb93a386Sopenharmony_ci trans[0] = matrix.getTranslateX(); 148cb93a386Sopenharmony_ci trans[1] = matrix.getTranslateY(); 149cb93a386Sopenharmony_ci this->append(SkRasterPipeline::matrix_translate, trans); 150cb93a386Sopenharmony_ci } else if ((mt | (SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask)) == 151cb93a386Sopenharmony_ci (SkMatrix::kScale_Mask | SkMatrix::kTranslate_Mask)) { 152cb93a386Sopenharmony_ci float* scaleTrans = alloc->makeArrayDefault<float>(4); 153cb93a386Sopenharmony_ci scaleTrans[0] = matrix.getScaleX(); 154cb93a386Sopenharmony_ci scaleTrans[1] = matrix.getScaleY(); 155cb93a386Sopenharmony_ci scaleTrans[2] = matrix.getTranslateX(); 156cb93a386Sopenharmony_ci scaleTrans[3] = matrix.getTranslateY(); 157cb93a386Sopenharmony_ci this->append(SkRasterPipeline::matrix_scale_translate, scaleTrans); 158cb93a386Sopenharmony_ci } else { 159cb93a386Sopenharmony_ci float* storage = alloc->makeArrayDefault<float>(9); 160cb93a386Sopenharmony_ci matrix.get9(storage); 161cb93a386Sopenharmony_ci if (!matrix.hasPerspective()) { 162cb93a386Sopenharmony_ci // note: asAffine and the 2x3 stage really only need 6 entries 163cb93a386Sopenharmony_ci this->append(SkRasterPipeline::matrix_2x3, storage); 164cb93a386Sopenharmony_ci } else { 165cb93a386Sopenharmony_ci this->append(SkRasterPipeline::matrix_perspective, storage); 166cb93a386Sopenharmony_ci } 167cb93a386Sopenharmony_ci } 168cb93a386Sopenharmony_ci} 169cb93a386Sopenharmony_ci 170cb93a386Sopenharmony_civoid SkRasterPipeline::append_load(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { 171cb93a386Sopenharmony_ci switch (ct) { 172cb93a386Sopenharmony_ci case kUnknown_SkColorType: SkASSERT(false); break; 173cb93a386Sopenharmony_ci 174cb93a386Sopenharmony_ci case kAlpha_8_SkColorType: this->append(load_a8, ctx); break; 175cb93a386Sopenharmony_ci case kA16_unorm_SkColorType: this->append(load_a16, ctx); break; 176cb93a386Sopenharmony_ci case kA16_float_SkColorType: this->append(load_af16, ctx); break; 177cb93a386Sopenharmony_ci case kRGB_565_SkColorType: this->append(load_565, ctx); break; 178cb93a386Sopenharmony_ci case kARGB_4444_SkColorType: this->append(load_4444, ctx); break; 179cb93a386Sopenharmony_ci case kR8G8_unorm_SkColorType: this->append(load_rg88, ctx); break; 180cb93a386Sopenharmony_ci case kR16G16_unorm_SkColorType: this->append(load_rg1616, ctx); break; 181cb93a386Sopenharmony_ci case kR16G16_float_SkColorType: this->append(load_rgf16, ctx); break; 182cb93a386Sopenharmony_ci case kRGBA_8888_SkColorType: this->append(load_8888, ctx); break; 183cb93a386Sopenharmony_ci case kRGBA_1010102_SkColorType: this->append(load_1010102, ctx); break; 184cb93a386Sopenharmony_ci case kR16G16B16A16_unorm_SkColorType:this->append(load_16161616,ctx); break; 185cb93a386Sopenharmony_ci case kRGBA_F16Norm_SkColorType: 186cb93a386Sopenharmony_ci case kRGBA_F16_SkColorType: this->append(load_f16, ctx); break; 187cb93a386Sopenharmony_ci case kRGBA_F32_SkColorType: this->append(load_f32, ctx); break; 188cb93a386Sopenharmony_ci 189cb93a386Sopenharmony_ci case kGray_8_SkColorType: this->append(load_a8, ctx); 190cb93a386Sopenharmony_ci this->append(alpha_to_gray); 191cb93a386Sopenharmony_ci break; 192cb93a386Sopenharmony_ci 193cb93a386Sopenharmony_ci case kRGB_888x_SkColorType: this->append(load_8888, ctx); 194cb93a386Sopenharmony_ci this->append(force_opaque); 195cb93a386Sopenharmony_ci break; 196cb93a386Sopenharmony_ci 197cb93a386Sopenharmony_ci case kBGRA_1010102_SkColorType: this->append(load_1010102, ctx); 198cb93a386Sopenharmony_ci this->append(swap_rb); 199cb93a386Sopenharmony_ci break; 200cb93a386Sopenharmony_ci 201cb93a386Sopenharmony_ci case kRGB_101010x_SkColorType: this->append(load_1010102, ctx); 202cb93a386Sopenharmony_ci this->append(force_opaque); 203cb93a386Sopenharmony_ci break; 204cb93a386Sopenharmony_ci 205cb93a386Sopenharmony_ci case kBGR_101010x_SkColorType: this->append(load_1010102, ctx); 206cb93a386Sopenharmony_ci this->append(force_opaque); 207cb93a386Sopenharmony_ci this->append(swap_rb); 208cb93a386Sopenharmony_ci break; 209cb93a386Sopenharmony_ci 210cb93a386Sopenharmony_ci case kBGRA_8888_SkColorType: this->append(load_8888, ctx); 211cb93a386Sopenharmony_ci this->append(swap_rb); 212cb93a386Sopenharmony_ci break; 213cb93a386Sopenharmony_ci 214cb93a386Sopenharmony_ci case kSRGBA_8888_SkColorType: 215cb93a386Sopenharmony_ci this->append(load_8888, ctx); 216cb93a386Sopenharmony_ci this->append_transfer_function(*skcms_sRGB_TransferFunction()); 217cb93a386Sopenharmony_ci break; 218cb93a386Sopenharmony_ci } 219cb93a386Sopenharmony_ci} 220cb93a386Sopenharmony_ci 221cb93a386Sopenharmony_civoid SkRasterPipeline::append_load_dst(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { 222cb93a386Sopenharmony_ci switch (ct) { 223cb93a386Sopenharmony_ci case kUnknown_SkColorType: SkASSERT(false); break; 224cb93a386Sopenharmony_ci 225cb93a386Sopenharmony_ci case kAlpha_8_SkColorType: this->append(load_a8_dst, ctx); break; 226cb93a386Sopenharmony_ci case kA16_unorm_SkColorType: this->append(load_a16_dst, ctx); break; 227cb93a386Sopenharmony_ci case kA16_float_SkColorType: this->append(load_af16_dst, ctx); break; 228cb93a386Sopenharmony_ci case kRGB_565_SkColorType: this->append(load_565_dst, ctx); break; 229cb93a386Sopenharmony_ci case kARGB_4444_SkColorType: this->append(load_4444_dst, ctx); break; 230cb93a386Sopenharmony_ci case kR8G8_unorm_SkColorType: this->append(load_rg88_dst, ctx); break; 231cb93a386Sopenharmony_ci case kR16G16_unorm_SkColorType: this->append(load_rg1616_dst, ctx); break; 232cb93a386Sopenharmony_ci case kR16G16_float_SkColorType: this->append(load_rgf16_dst, ctx); break; 233cb93a386Sopenharmony_ci case kRGBA_8888_SkColorType: this->append(load_8888_dst, ctx); break; 234cb93a386Sopenharmony_ci case kRGBA_1010102_SkColorType: this->append(load_1010102_dst, ctx); break; 235cb93a386Sopenharmony_ci case kR16G16B16A16_unorm_SkColorType: this->append(load_16161616_dst,ctx); break; 236cb93a386Sopenharmony_ci case kRGBA_F16Norm_SkColorType: 237cb93a386Sopenharmony_ci case kRGBA_F16_SkColorType: this->append(load_f16_dst, ctx); break; 238cb93a386Sopenharmony_ci case kRGBA_F32_SkColorType: this->append(load_f32_dst, ctx); break; 239cb93a386Sopenharmony_ci 240cb93a386Sopenharmony_ci case kGray_8_SkColorType: this->append(load_a8_dst, ctx); 241cb93a386Sopenharmony_ci this->append(alpha_to_gray_dst); 242cb93a386Sopenharmony_ci break; 243cb93a386Sopenharmony_ci 244cb93a386Sopenharmony_ci case kRGB_888x_SkColorType: this->append(load_8888_dst, ctx); 245cb93a386Sopenharmony_ci this->append(force_opaque_dst); 246cb93a386Sopenharmony_ci break; 247cb93a386Sopenharmony_ci 248cb93a386Sopenharmony_ci case kBGRA_1010102_SkColorType: this->append(load_1010102_dst, ctx); 249cb93a386Sopenharmony_ci this->append(swap_rb_dst); 250cb93a386Sopenharmony_ci break; 251cb93a386Sopenharmony_ci 252cb93a386Sopenharmony_ci case kRGB_101010x_SkColorType: this->append(load_1010102_dst, ctx); 253cb93a386Sopenharmony_ci this->append(force_opaque_dst); 254cb93a386Sopenharmony_ci break; 255cb93a386Sopenharmony_ci 256cb93a386Sopenharmony_ci case kBGR_101010x_SkColorType: this->append(load_1010102_dst, ctx); 257cb93a386Sopenharmony_ci this->append(force_opaque_dst); 258cb93a386Sopenharmony_ci this->append(swap_rb_dst); 259cb93a386Sopenharmony_ci break; 260cb93a386Sopenharmony_ci 261cb93a386Sopenharmony_ci case kBGRA_8888_SkColorType: this->append(load_8888_dst, ctx); 262cb93a386Sopenharmony_ci this->append(swap_rb_dst); 263cb93a386Sopenharmony_ci break; 264cb93a386Sopenharmony_ci 265cb93a386Sopenharmony_ci case kSRGBA_8888_SkColorType: 266cb93a386Sopenharmony_ci // TODO: We could remove the double-swap if we had _dst versions of all the TF stages 267cb93a386Sopenharmony_ci this->append(load_8888_dst, ctx); 268cb93a386Sopenharmony_ci this->append(swap_src_dst); 269cb93a386Sopenharmony_ci this->append_transfer_function(*skcms_sRGB_TransferFunction()); 270cb93a386Sopenharmony_ci this->append(swap_src_dst); 271cb93a386Sopenharmony_ci break; 272cb93a386Sopenharmony_ci } 273cb93a386Sopenharmony_ci} 274cb93a386Sopenharmony_ci 275cb93a386Sopenharmony_civoid SkRasterPipeline::append_store(SkColorType ct, const SkRasterPipeline_MemoryCtx* ctx) { 276cb93a386Sopenharmony_ci switch (ct) { 277cb93a386Sopenharmony_ci case kUnknown_SkColorType: SkASSERT(false); break; 278cb93a386Sopenharmony_ci 279cb93a386Sopenharmony_ci case kAlpha_8_SkColorType: this->append(store_a8, ctx); break; 280cb93a386Sopenharmony_ci case kA16_unorm_SkColorType: this->append(store_a16, ctx); break; 281cb93a386Sopenharmony_ci case kA16_float_SkColorType: this->append(store_af16, ctx); break; 282cb93a386Sopenharmony_ci case kRGB_565_SkColorType: this->append(store_565, ctx); break; 283cb93a386Sopenharmony_ci case kARGB_4444_SkColorType: this->append(store_4444, ctx); break; 284cb93a386Sopenharmony_ci case kR8G8_unorm_SkColorType: this->append(store_rg88, ctx); break; 285cb93a386Sopenharmony_ci case kR16G16_unorm_SkColorType: this->append(store_rg1616, ctx); break; 286cb93a386Sopenharmony_ci case kR16G16_float_SkColorType: this->append(store_rgf16, ctx); break; 287cb93a386Sopenharmony_ci case kRGBA_8888_SkColorType: this->append(store_8888, ctx); break; 288cb93a386Sopenharmony_ci case kRGBA_1010102_SkColorType: this->append(store_1010102, ctx); break; 289cb93a386Sopenharmony_ci case kR16G16B16A16_unorm_SkColorType: this->append(store_16161616,ctx); break; 290cb93a386Sopenharmony_ci case kRGBA_F16Norm_SkColorType: 291cb93a386Sopenharmony_ci case kRGBA_F16_SkColorType: this->append(store_f16, ctx); break; 292cb93a386Sopenharmony_ci case kRGBA_F32_SkColorType: this->append(store_f32, ctx); break; 293cb93a386Sopenharmony_ci 294cb93a386Sopenharmony_ci case kRGB_888x_SkColorType: this->append(force_opaque); 295cb93a386Sopenharmony_ci this->append(store_8888, ctx); 296cb93a386Sopenharmony_ci break; 297cb93a386Sopenharmony_ci 298cb93a386Sopenharmony_ci case kBGRA_1010102_SkColorType: this->append(swap_rb); 299cb93a386Sopenharmony_ci this->append(store_1010102, ctx); 300cb93a386Sopenharmony_ci break; 301cb93a386Sopenharmony_ci 302cb93a386Sopenharmony_ci case kRGB_101010x_SkColorType: this->append(force_opaque); 303cb93a386Sopenharmony_ci this->append(store_1010102, ctx); 304cb93a386Sopenharmony_ci break; 305cb93a386Sopenharmony_ci 306cb93a386Sopenharmony_ci case kBGR_101010x_SkColorType: this->append(force_opaque); 307cb93a386Sopenharmony_ci this->append(swap_rb); 308cb93a386Sopenharmony_ci this->append(store_1010102, ctx); 309cb93a386Sopenharmony_ci break; 310cb93a386Sopenharmony_ci 311cb93a386Sopenharmony_ci case kGray_8_SkColorType: this->append(bt709_luminance_or_luma_to_alpha); 312cb93a386Sopenharmony_ci this->append(store_a8, ctx); 313cb93a386Sopenharmony_ci break; 314cb93a386Sopenharmony_ci 315cb93a386Sopenharmony_ci case kBGRA_8888_SkColorType: this->append(swap_rb); 316cb93a386Sopenharmony_ci this->append(store_8888, ctx); 317cb93a386Sopenharmony_ci break; 318cb93a386Sopenharmony_ci 319cb93a386Sopenharmony_ci case kSRGBA_8888_SkColorType: 320cb93a386Sopenharmony_ci this->append_transfer_function(*skcms_sRGB_Inverse_TransferFunction()); 321cb93a386Sopenharmony_ci this->append(store_8888, ctx); 322cb93a386Sopenharmony_ci break; 323cb93a386Sopenharmony_ci } 324cb93a386Sopenharmony_ci} 325cb93a386Sopenharmony_ci 326cb93a386Sopenharmony_civoid SkRasterPipeline::append_transfer_function(const skcms_TransferFunction& tf) { 327cb93a386Sopenharmony_ci void* ctx = const_cast<void*>(static_cast<const void*>(&tf)); 328cb93a386Sopenharmony_ci switch (classify_transfer_fn(tf)) { 329cb93a386Sopenharmony_ci case Bad_TF: SkASSERT(false); break; 330cb93a386Sopenharmony_ci 331cb93a386Sopenharmony_ci case TFKind::sRGBish_TF: 332cb93a386Sopenharmony_ci if (tf.a == 1 && tf.b == 0 && tf.c == 0 && tf.d == 0 && tf.e == 0 && tf.f == 0) { 333cb93a386Sopenharmony_ci this->unchecked_append(gamma_, ctx); 334cb93a386Sopenharmony_ci } else { 335cb93a386Sopenharmony_ci this->unchecked_append(parametric, ctx); 336cb93a386Sopenharmony_ci } 337cb93a386Sopenharmony_ci break; 338cb93a386Sopenharmony_ci case PQish_TF: this->unchecked_append(PQish, ctx); break; 339cb93a386Sopenharmony_ci case HLGish_TF: this->unchecked_append(HLGish, ctx); break; 340cb93a386Sopenharmony_ci case HLGinvish_TF: this->unchecked_append(HLGinvish, ctx); break; 341cb93a386Sopenharmony_ci } 342cb93a386Sopenharmony_ci} 343cb93a386Sopenharmony_ci 344cb93a386Sopenharmony_ci// Clamp premul values to [0,alpha] (logical [0,1]) to avoid the confusing 345cb93a386Sopenharmony_ci// scenario of being able to store a logical color channel > 1.0 when alpha < 1.0. 346cb93a386Sopenharmony_ci// Most software that works with normalized premul values expect r,g,b channels all <= a. 347cb93a386Sopenharmony_ci// 348cb93a386Sopenharmony_ci// In addition, GL clamps all its color channels to limits of the format just 349cb93a386Sopenharmony_ci// before the blend step (~here). To match that auto-clamp, we clamp alpha to 350cb93a386Sopenharmony_ci// [0,1] too, just in case someone gave us a crazy alpha. 351cb93a386Sopenharmony_civoid SkRasterPipeline::append_gamut_clamp_if_normalized(const SkImageInfo& info) { 352cb93a386Sopenharmony_ci if (info.alphaType() == kPremul_SkAlphaType && SkColorTypeIsNormalized(info.colorType())) { 353cb93a386Sopenharmony_ci this->unchecked_append(SkRasterPipeline::clamp_gamut, nullptr); 354cb93a386Sopenharmony_ci } 355cb93a386Sopenharmony_ci} 356cb93a386Sopenharmony_ci 357cb93a386Sopenharmony_ciSkRasterPipeline::StartPipelineFn SkRasterPipeline::build_pipeline(void** ip) const { 358cb93a386Sopenharmony_ci if (!gForceHighPrecisionRasterPipeline) { 359cb93a386Sopenharmony_ci // We'll try to build a lowp pipeline, but if that fails fallback to a highp float pipeline. 360cb93a386Sopenharmony_ci void** reset_point = ip; 361cb93a386Sopenharmony_ci 362cb93a386Sopenharmony_ci // Stages are stored backwards in fStages, so we reverse here, back to front. 363cb93a386Sopenharmony_ci *--ip = (void*)SkOpts::just_return_lowp; 364cb93a386Sopenharmony_ci for (const StageList* st = fStages; st; st = st->prev) { 365cb93a386Sopenharmony_ci if (auto fn = SkOpts::stages_lowp[st->stage]) { 366cb93a386Sopenharmony_ci if (st->ctx) { 367cb93a386Sopenharmony_ci *--ip = st->ctx; 368cb93a386Sopenharmony_ci } 369cb93a386Sopenharmony_ci *--ip = (void*)fn; 370cb93a386Sopenharmony_ci } else { 371cb93a386Sopenharmony_ci ip = reset_point; 372cb93a386Sopenharmony_ci break; 373cb93a386Sopenharmony_ci } 374cb93a386Sopenharmony_ci } 375cb93a386Sopenharmony_ci if (ip != reset_point) { 376cb93a386Sopenharmony_ci return SkOpts::start_pipeline_lowp; 377cb93a386Sopenharmony_ci } 378cb93a386Sopenharmony_ci } 379cb93a386Sopenharmony_ci 380cb93a386Sopenharmony_ci *--ip = (void*)SkOpts::just_return_highp; 381cb93a386Sopenharmony_ci for (const StageList* st = fStages; st; st = st->prev) { 382cb93a386Sopenharmony_ci if (st->ctx) { 383cb93a386Sopenharmony_ci *--ip = st->ctx; 384cb93a386Sopenharmony_ci } 385cb93a386Sopenharmony_ci *--ip = (void*)SkOpts::stages_highp[st->stage]; 386cb93a386Sopenharmony_ci } 387cb93a386Sopenharmony_ci return SkOpts::start_pipeline_highp; 388cb93a386Sopenharmony_ci} 389cb93a386Sopenharmony_ci 390cb93a386Sopenharmony_civoid SkRasterPipeline::run(size_t x, size_t y, size_t w, size_t h) const { 391cb93a386Sopenharmony_ci if (this->empty()) { 392cb93a386Sopenharmony_ci return; 393cb93a386Sopenharmony_ci } 394cb93a386Sopenharmony_ci 395cb93a386Sopenharmony_ci // Best to not use fAlloc here... we can't bound how often run() will be called. 396cb93a386Sopenharmony_ci SkAutoSTMalloc<64, void*> program(fSlotsNeeded); 397cb93a386Sopenharmony_ci 398cb93a386Sopenharmony_ci auto start_pipeline = this->build_pipeline(program.get() + fSlotsNeeded); 399cb93a386Sopenharmony_ci start_pipeline(x,y,x+w,y+h, program.get()); 400cb93a386Sopenharmony_ci} 401cb93a386Sopenharmony_ci 402cb93a386Sopenharmony_cistd::function<void(size_t, size_t, size_t, size_t)> SkRasterPipeline::compile() const { 403cb93a386Sopenharmony_ci if (this->empty()) { 404cb93a386Sopenharmony_ci return [](size_t, size_t, size_t, size_t) {}; 405cb93a386Sopenharmony_ci } 406cb93a386Sopenharmony_ci 407cb93a386Sopenharmony_ci void** program = fAlloc->makeArray<void*>(fSlotsNeeded); 408cb93a386Sopenharmony_ci 409cb93a386Sopenharmony_ci auto start_pipeline = this->build_pipeline(program + fSlotsNeeded); 410cb93a386Sopenharmony_ci return [=](size_t x, size_t y, size_t w, size_t h) { 411cb93a386Sopenharmony_ci start_pipeline(x,y,x+w,y+h, program); 412cb93a386Sopenharmony_ci }; 413cb93a386Sopenharmony_ci} 414