Coverage Report

Created: 2024-09-14 07:19

/src/skia/tools/gpu/TestOps.cpp
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright 2019 Google LLC
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 "tools/gpu/TestOps.h"
9
10
#include "src/core/SkPointPriv.h"
11
#include "src/gpu/BufferWriter.h"
12
#include "src/gpu/KeyBuilder.h"
13
#include "src/gpu/ganesh/GrCaps.h"
14
#include "src/gpu/ganesh/GrGeometryProcessor.h"
15
#include "src/gpu/ganesh/GrMemoryPool.h"
16
#include "src/gpu/ganesh/GrOpFlushState.h"
17
#include "src/gpu/ganesh/GrProgramInfo.h"
18
#include "src/gpu/ganesh/glsl/GrGLSLFragmentShaderBuilder.h"
19
#include "src/gpu/ganesh/glsl/GrGLSLVarying.h"
20
#include "src/gpu/ganesh/glsl/GrGLSLVertexGeoBuilder.h"
21
#include "src/gpu/ganesh/ops/GrMeshDrawOp.h"
22
#include "src/gpu/ganesh/ops/GrSimpleMeshDrawOpHelper.h"
23
24
namespace {
25
26
class GP : public GrGeometryProcessor {
27
public:
28
    GP(const SkMatrix& localMatrix, bool wideColor)
29
0
            : GrGeometryProcessor(kTestRectOp_ClassID), fLocalMatrix(localMatrix) {
30
0
        fInColor = MakeColorAttribute("color", wideColor);
31
0
        this->setVertexAttributesWithImplicitOffsets(&fInPosition, 3);
32
0
    }
33
34
0
    const char* name() const override { return "TestRectOp::GP"; }
35
36
0
    std::unique_ptr<ProgramImpl> makeProgramImpl(const GrShaderCaps&) const override {
37
0
        class Impl : public ProgramImpl {
38
0
        public:
39
0
            void setData(const GrGLSLProgramDataManager& pdman,
40
0
                         const GrShaderCaps& shaderCaps,
41
0
                         const GrGeometryProcessor& geomProc) override {
42
0
                const auto& gp = geomProc.cast<GP>();
43
0
                SetTransform(pdman, shaderCaps, fLocalMatrixUni, gp.fLocalMatrix);
44
0
            }
45
46
0
        private:
47
0
            void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
48
0
                const auto& gp = args.fGeomProc.cast<GP>();
49
0
                args.fVaryingHandler->emitAttributes(gp);
50
0
                GrGLSLVarying colorVarying(SkSLType::kHalf4);
51
0
                args.fVaryingHandler->addVarying("color", &colorVarying,
52
0
                                                 GrGLSLVaryingHandler::Interpolation::kCanBeFlat);
53
0
                args.fVertBuilder->codeAppendf("%s = %s;", colorVarying.vsOut(), gp.fInColor.name());
54
0
                args.fFragBuilder->codeAppendf("half4 %s = %s;",
55
0
                                               args.fOutputColor, colorVarying.fsIn());
56
0
                args.fFragBuilder->codeAppendf("const half4 %s = half4(1);", args.fOutputCoverage);
57
0
                WriteOutputPosition(args.fVertBuilder, gpArgs, gp.fInPosition.name());
58
0
                WriteLocalCoord(args.fVertBuilder,
59
0
                                args.fUniformHandler,
60
0
                                *args.fShaderCaps,
61
0
                                gpArgs,
62
0
                                gp.fInLocalCoords.asShaderVar(),
63
0
                                gp.fLocalMatrix,
64
0
                                &fLocalMatrixUni);
65
0
            }
66
67
0
            UniformHandle fLocalMatrixUni;
68
0
        };
69
70
0
        return std::make_unique<Impl>();
71
0
    }
72
73
0
    void addToKey(const GrShaderCaps& shaderCaps, skgpu::KeyBuilder* b) const override {
74
0
        b->add32(ProgramImpl::ComputeMatrixKey(shaderCaps, fLocalMatrix));
75
0
    }
76
77
0
    bool wideColor() const { return fInColor.cpuType() != kUByte4_norm_GrVertexAttribType; }
78
79
private:
80
    Attribute fInPosition    = {   "inPosition", kFloat2_GrVertexAttribType, SkSLType::kFloat2};
81
    Attribute fInLocalCoords = {"inLocalCoords", kFloat2_GrVertexAttribType, SkSLType::kFloat2};
82
    Attribute fInColor;
83
84
    SkMatrix fLocalMatrix;
85
};
86
87
class TestRectOp final : public GrMeshDrawOp {
88
public:
89
    static GrOp::Owner Make(GrRecordingContext*,
90
                            GrPaint&&,
91
                            const SkRect& drawRect,
92
                            const SkRect& localRect,
93
                            const SkMatrix& localM);
94
95
0
    const char* name() const override { return "TestRectOp"; }
96
97
0
    FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; }
98
99
    GrProcessorSet::Analysis finalize(const GrCaps&,
100
                                      const GrAppliedClip*,
101
                                      GrClampType) override;
102
103
0
    void visitProxies(const GrVisitProxyFunc& func) const override {
104
0
        if (fProgramInfo) {
105
0
            fProgramInfo->visitFPProxies(func);
106
0
        } else {
107
0
            fProcessorSet.visitProxies(func);
108
0
        }
109
0
    }
110
111
private:
112
    DEFINE_OP_CLASS_ID
113
114
    TestRectOp(const GrCaps*,
115
               GrPaint&&,
116
               const SkRect& drawRect,
117
               const SkRect& localRect,
118
               const SkMatrix& localMatrix);
119
120
0
    GrProgramInfo* programInfo() override { return fProgramInfo; }
121
    void onCreateProgramInfo(const GrCaps*,
122
                             SkArenaAlloc*,
123
                             const GrSurfaceProxyView& writeView,
124
                             bool usesMSAASurface,
125
                             GrAppliedClip&&,
126
                             const GrDstProxyView&,
127
                             GrXferBarrierFlags renderPassXferBarriers,
128
                             GrLoadOp colorLoadOp) override;
129
130
    void onPrepareDraws(GrMeshDrawTarget*) override;
131
    void onExecute(GrOpFlushState*, const SkRect& chainBounds) override;
132
133
    SkRect         fDrawRect;
134
    SkRect         fLocalRect;
135
    SkPMColor4f    fColor;
136
    GP             fGP;
137
    GrProcessorSet fProcessorSet;
138
139
    // If this op is prePrepared the created programInfo will be stored here for use in
140
    // onExecute. In the prePrepared case it will have been stored in the record-time arena.
141
    GrProgramInfo* fProgramInfo = nullptr;
142
    GrSimpleMesh*  fMesh        = nullptr;
143
144
    friend class ::GrOp;
145
};
146
147
GrOp::Owner TestRectOp::Make(GrRecordingContext* context,
148
                             GrPaint&& paint,
149
                             const SkRect& drawRect,
150
                             const SkRect& localRect,
151
0
                             const SkMatrix& localM) {
152
0
    const auto* caps = context->priv().caps();
153
0
    return GrOp::Make<TestRectOp>(context, caps, std::move(paint), drawRect, localRect, localM);
154
0
}
155
156
GrProcessorSet::Analysis TestRectOp::finalize(const GrCaps& caps,
157
                                              const GrAppliedClip* clip,
158
0
                                              GrClampType clampType) {
159
0
    return fProcessorSet.finalize(GrProcessorAnalysisColor::Opaque::kYes,
160
0
                                  GrProcessorAnalysisCoverage::kSingleChannel, clip,
161
0
                                  &GrUserStencilSettings::kUnused, caps, clampType, &fColor);
162
0
}
163
164
0
static bool use_wide_color(const GrPaint& paint, const GrCaps* caps) {
165
0
    return !paint.getColor4f().fitsInBytes() && caps->halfFloatVertexAttributeSupport();
166
0
}
167
TestRectOp::TestRectOp(const GrCaps* caps,
168
                       GrPaint&& paint,
169
                       const SkRect& drawRect,
170
                       const SkRect& localRect,
171
                       const SkMatrix& localMatrix)
172
        : GrMeshDrawOp(ClassID())
173
        , fDrawRect(drawRect)
174
        , fLocalRect(localRect)
175
        , fColor(paint.getColor4f())
176
        , fGP(localMatrix, use_wide_color(paint, caps))
177
0
        , fProcessorSet(std::move(paint)) {
178
0
    this->setBounds(drawRect.makeSorted(), HasAABloat::kNo, IsHairline::kNo);
179
0
}
180
181
void TestRectOp::onCreateProgramInfo(const GrCaps* caps,
182
                                     SkArenaAlloc* arena,
183
                                     const GrSurfaceProxyView& writeView,
184
                                     bool usesMSAASurface,
185
                                     GrAppliedClip&& appliedClip,
186
                                     const GrDstProxyView& dstProxyView,
187
                                     GrXferBarrierFlags renderPassXferBarriers,
188
0
                                     GrLoadOp colorLoadOp) {
189
0
    fProgramInfo = GrSimpleMeshDrawOpHelper::CreateProgramInfo(caps,
190
0
                                                               arena,
191
0
                                                               writeView,
192
0
                                                               usesMSAASurface,
193
0
                                                               std::move(appliedClip),
194
0
                                                               dstProxyView,
195
0
                                                               &fGP,
196
0
                                                               std::move(fProcessorSet),
197
0
                                                               GrPrimitiveType::kTriangles,
198
0
                                                               renderPassXferBarriers,
199
0
                                                               colorLoadOp,
200
0
                                                               GrPipeline::InputFlags::kNone);
201
0
}
202
203
0
void TestRectOp::onPrepareDraws(GrMeshDrawTarget* target) {
204
0
    QuadHelper helper(target, fGP.vertexStride(), 1);
205
0
    skgpu::VertexWriter writer{helper.vertices()};
206
0
    auto pos = skgpu::VertexWriter::TriStripFromRect(fDrawRect);
207
0
    auto local = skgpu::VertexWriter::TriStripFromRect(fLocalRect);
208
0
    skgpu::VertexColor color(fColor, fGP.wideColor());
209
0
    writer.writeQuad(pos, local, color);
210
211
0
    fMesh = helper.mesh();
212
0
}
213
214
0
void TestRectOp::onExecute(GrOpFlushState* flushState, const SkRect& chainBounds) {
215
0
    if (!fProgramInfo) {
216
0
        this->createProgramInfo(flushState);
217
0
    }
218
219
0
    flushState->bindPipelineAndScissorClip(*fProgramInfo, chainBounds);
220
0
    flushState->bindTextures(fProgramInfo->geomProc(), nullptr, fProgramInfo->pipeline());
221
0
    flushState->drawMesh(*fMesh);
222
0
}
223
224
}  // anonymous namespace
225
226
namespace sk_gpu_test::test_ops {
227
228
GrOp::Owner MakeRect(GrRecordingContext* context,
229
                     GrPaint&& paint,
230
                     const SkRect& drawRect,
231
                     const SkRect& localRect,
232
0
                     const SkMatrix& localM) {
233
0
    return TestRectOp::Make(context, std::move(paint), drawRect, localRect, localM);
234
0
}
235
236
GrOp::Owner MakeRect(GrRecordingContext* context,
237
                     std::unique_ptr<GrFragmentProcessor> fp,
238
                     const SkRect& drawRect,
239
                     const SkRect& localRect,
240
0
                     const SkMatrix& localM) {
241
0
    GrPaint paint;
242
0
    paint.setColorFragmentProcessor(std::move(fp));
243
0
    return TestRectOp::Make(context, std::move(paint), drawRect, localRect, localM);
244
0
}
245
246
GrOp::Owner MakeRect(GrRecordingContext* context,
247
                     GrPaint&& paint,
248
0
                     const SkRect& rect) {
249
0
    return TestRectOp::Make(context, std::move(paint), rect, rect, SkMatrix::I());
250
0
}
251
252
}  // namespace sk_gpu_test::test_ops