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