blob: d285a0d97a9bd2d28645dcc34d3e8997e4fc8495 [file] [log] [blame]
John Stiles6f3015a2020-10-08 14:55:36 -04001/*
2 * Copyright 2020 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 "tests/Test.h"
9
Brian Salomon48959462021-08-11 13:01:06 -040010#include "src/gpu/GrFragmentProcessor.h"
Robert Phillips550de7f2021-07-06 16:28:52 -040011#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
Robert Phillips4dca8312021-07-28 15:13:20 -040012#include "src/gpu/v1/SurfaceDrawContext_v1.h"
John Stiles6f3015a2020-10-08 14:55:36 -040013
Brian Salomoneebe7352020-12-09 16:37:04 -050014static void run_test(skiatest::Reporter*, GrDirectContext*,
Robert Phillips4dca8312021-07-28 15:13:20 -040015 skgpu::v1::SurfaceDrawContext*, SkVector a,
John Stiles6f3015a2020-10-08 14:55:36 -040016 SkVector b, float expectedCrossProduct);
17
18// This is a GPU test that ensures the SkSL 2d cross() intrinsic returns the correct sign (negative,
19// positive, or zero).
20DEF_GPUTEST_FOR_RENDERING_CONTEXTS(SkSLCross, reporter, ctxInfo) {
Robert Phillips4dca8312021-07-28 15:13:20 -040021 GrDirectContext* dContext = ctxInfo.directContext();
22 auto sdc = skgpu::v1::SurfaceDrawContext::Make(dContext, GrColorType::kRGBA_8888, nullptr,
23 SkBackingFit::kExact, {1, 1}, SkSurfaceProps());
24 if (!sdc) {
John Stiles6f3015a2020-10-08 14:55:36 -040025 ERRORF(reporter, "could not create render target context.");
26 return;
27 }
Robert Phillips4dca8312021-07-28 15:13:20 -040028 run_test(reporter, dContext, sdc.get(), {3,4}, {5,6}, -2); // Negative.
29 run_test(reporter, dContext, sdc.get(), {3,4}, {-5,-6}, 2); // Positive.
30 run_test(reporter, dContext, sdc.get(), {0, 2.287f}, {0, -7.741f}, 0); // Zero.
31 run_test(reporter, dContext, sdc.get(), {62.17f, 0}, {-43.49f, 0}, 0); // Zero.
John Stiles6f3015a2020-10-08 14:55:36 -040032}
33
34namespace {
35
36// Outputs:
37// Green if cross(a,b) > 0
38// Red if cross(a,b) < 0
39// Black if cross(a,b) == 0
40class VisualizeCrossProductSignFP : public GrFragmentProcessor {
41public:
42 VisualizeCrossProductSignFP(SkVector a, SkVector b)
43 : GrFragmentProcessor(kTestFP_ClassID, kPreservesOpaqueInput_OptimizationFlag)
44 , fA(a), fB(b) {
45 }
46
John Stiles6f3015a2020-10-08 14:55:36 -040047 const char* name() const override { return "VisualizeCrossProductSignFP"; }
Brian Salomonb25560a2021-08-10 13:56:13 -040048
John Stiles6f3015a2020-10-08 14:55:36 -040049 std::unique_ptr<GrFragmentProcessor> clone() const override {
50 return std::unique_ptr<GrFragmentProcessor>(new VisualizeCrossProductSignFP(fA, fB));
51 }
Brian Salomonb25560a2021-08-10 13:56:13 -040052
53private:
Brian Salomon13b28732021-08-06 15:33:58 -040054 void onAddToKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override {}
John Stiles6f3015a2020-10-08 14:55:36 -040055 bool onIsEqual(const GrFragmentProcessor&) const override { return true; }
56
Brian Salomonb25560a2021-08-10 13:56:13 -040057 std::unique_ptr<ProgramImpl> onMakeProgramImpl() const override {
58 class Impl : public ProgramImpl {
59 public:
60 void emitCode(EmitArgs& args) override {
61 auto& fp = args.fFp.cast<VisualizeCrossProductSignFP>();
62 const char *a, *b;
63 fAUniform = args.fUniformHandler->addUniform(&fp, kFragment_GrShaderFlag,
64 GrSLType::kFloat2_GrSLType, "a", &a);
65 fBUniform = args.fUniformHandler->addUniform(&fp, kFragment_GrShaderFlag,
66 GrSLType::kFloat2_GrSLType, "b", &b);
67 args.fFragBuilder->codeAppendf(R"(
John Stiles6f3015a2020-10-08 14:55:36 -040068 float crossProduct = cross(%s, %s);
69 float2 visualization = clamp(float2(-sign(crossProduct), sign(crossProduct)),
70 float2(0), float2(1));
Brian Salomonb25560a2021-08-10 13:56:13 -040071 return half2(visualization).xy01;)", a, b);
72 }
John Stiles6f3015a2020-10-08 14:55:36 -040073
Brian Salomonb25560a2021-08-10 13:56:13 -040074 private:
75 void onSetData(const GrGLSLProgramDataManager& pdman,
76 const GrFragmentProcessor& processor) override {
77 const auto& fp = processor.cast<VisualizeCrossProductSignFP>();
78 pdman.set2f(fAUniform, fp.fA.x(), fp.fA.y());
79 pdman.set2f(fBUniform, fp.fB.x(), fp.fB.y());
80 }
81 GrGLSLUniformHandler::UniformHandle fAUniform;
82 GrGLSLUniformHandler::UniformHandle fBUniform;
83 };
84
Brian Salomon18ab2032021-02-23 10:07:05 -050085 return std::make_unique<Impl>();
86 }
John Stiles6f3015a2020-10-08 14:55:36 -040087 const SkVector fA, fB;
88};
89
90} // namespace
91
92static void run_test(skiatest::Reporter* reporter, GrDirectContext* directContext,
Robert Phillips4dca8312021-07-28 15:13:20 -040093 skgpu::v1::SurfaceDrawContext* sdc, SkVector a, SkVector b,
John Stiles6f3015a2020-10-08 14:55:36 -040094 float expectedCrossProduct) {
Robert Phillips4dca8312021-07-28 15:13:20 -040095 SkASSERT(sdc->width() == 1);
96 SkASSERT(sdc->height() == 1);
John Stiles6f3015a2020-10-08 14:55:36 -040097
Robert Phillips4dca8312021-07-28 15:13:20 -040098 sdc->clear(SkPMColor4f::FromBytes_RGBA(0xbaaaaaad));
John Stiles6f3015a2020-10-08 14:55:36 -040099
100 GrPaint crossPaint;
101 crossPaint.setColor4f(SK_PMColor4fWHITE);
102 crossPaint.setPorterDuffXPFactory(SkBlendMode::kSrcOver);
103 crossPaint.setColorFragmentProcessor(std::make_unique<VisualizeCrossProductSignFP>(a, b));
Robert Phillips4dca8312021-07-28 15:13:20 -0400104 sdc->drawRect(/*clip=*/nullptr, std::move(crossPaint), GrAA::kNo, SkMatrix::I(),
John Stiles6f3015a2020-10-08 14:55:36 -0400105 SkRect::MakeWH(1,1));
106
107 GrColor result;
Brian Salomondd4087d2020-12-23 20:36:44 -0500108 GrPixmap resultPM(SkImageInfo::Make(1, 1, kRGBA_8888_SkColorType, kPremul_SkAlphaType),
109 &result,
110 sizeof(GrColor));
Robert Phillips4dca8312021-07-28 15:13:20 -0400111 sdc->readPixels(directContext, resultPM, {0, 0});
John Stiles6f3015a2020-10-08 14:55:36 -0400112
113 SkASSERT(expectedCrossProduct == a.cross(b));
114 if (expectedCrossProduct > 0) {
115 REPORTER_ASSERT(reporter, result == GrColorPackRGBA(0, 255, 0, 255)); // Green.
116 } else if (expectedCrossProduct < 0) {
117 REPORTER_ASSERT(reporter, result == GrColorPackRGBA(255, 0, 0, 255)); // Red.
118 } else {
119 REPORTER_ASSERT(reporter, result == GrColorPackRGBA(0, 0, 0, 255)); // Black.
120 }
121}