skia2/tools/gpu/TestOps.cpp

252 lines
10 KiB
C++
Raw Normal View History

/*
* Copyright 2019 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "tools/gpu/TestOps.h"
#include "src/core/SkPointPriv.h"
#include "src/gpu/BufferWriter.h"
#include "src/gpu/GrCaps.h"
#include "src/gpu/GrGeometryProcessor.h"
#include "src/gpu/GrMemoryPool.h"
#include "src/gpu/GrOpFlushState.h"
#include "src/gpu/GrProgramInfo.h"
#include "src/gpu/KeyBuilder.h"
#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
#include "src/gpu/glsl/GrGLSLVarying.h"
#include "src/gpu/glsl/GrGLSLVertexGeoBuilder.h"
#include "src/gpu/ops/GrSimpleMeshDrawOpHelper.h"
namespace {
class GP : public GrGeometryProcessor {
public:
GP(const SkMatrix& localMatrix, bool wideColor)
: GrGeometryProcessor(kTestRectOp_ClassID), fLocalMatrix(localMatrix) {
fInColor = MakeColorAttribute("color", wideColor);
this->setVertexAttributesWithImplicitOffsets(&fInPosition, 3);
}
const char* name() const override { return "TestRectOp::GP"; }
std::unique_ptr<ProgramImpl> makeProgramImpl(const GrShaderCaps&) const override {
class Impl : public ProgramImpl {
public:
void setData(const GrGLSLProgramDataManager& pdman,
const GrShaderCaps& shaderCaps,
const GrGeometryProcessor& geomProc) override {
const auto& gp = geomProc.cast<GP>();
SetTransform(pdman, shaderCaps, fLocalMatrixUni, gp.fLocalMatrix);
}
private:
void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
const auto& gp = args.fGeomProc.cast<GP>();
args.fVaryingHandler->emitAttributes(gp);
GrGLSLVarying colorVarying(kHalf4_GrSLType);
args.fVaryingHandler->addVarying("color", &colorVarying,
GrGLSLVaryingHandler::Interpolation::kCanBeFlat);
args.fVertBuilder->codeAppendf("%s = %s;", colorVarying.vsOut(), gp.fInColor.name());
args.fFragBuilder->codeAppendf("half4 %s = %s;",
args.fOutputColor, colorVarying.fsIn());
args.fFragBuilder->codeAppendf("const half4 %s = half4(1);", args.fOutputCoverage);
WriteOutputPosition(args.fVertBuilder, gpArgs, gp.fInPosition.name());
WriteLocalCoord(args.fVertBuilder,
args.fUniformHandler,
*args.fShaderCaps,
gpArgs,
gp.fInLocalCoords.asShaderVar(),
gp.fLocalMatrix,
&fLocalMatrixUni);
}
UniformHandle fLocalMatrixUni;
};
return std::make_unique<Impl>();
Reland "Centralize geometry processor vertex shader transform code" This is a reland of 0426947243e1d6b470830c4d3b1c5704ed1f23e1 Original change's description: > Centralize geometry processor vertex shader transform code > > GrGLSLGeometryProcessors no longer have to call emitTransforms() in > their onEmitCode() function. Instead, the GpArgs struct allows them to > set a GrShaderVar that holds the computed or explicitly provided local > coordinates in the vertex shader. > > The base GrGLSLGeometryProcessor now automatically uses that to collect > all of the transforms that can then be lifted out of FPs to the vertex > shader, and base their computation on the GP provided local coordinate. > > As part of this, there is no more built-in magic concatenation of a > local matrix / inverse view matrix to these coordinate transforms. GP > implementations that relied on this now manage their own uniform for this > matrix and compute the local coordinate before assigning to GpArgs. > > The base GrGLSLGeometryProcessor is updated to provide helpers for this > pattern. > > Bug: skia:10396 > Change-Id: I56afb3fff4b806f6015ab13626ac1afde9ef5c2b > Reviewed-on: https://skia-review.googlesource.com/c/skia/+/297027 > Commit-Queue: Michael Ludwig <michaelludwig@google.com> > Reviewed-by: Brian Osman <brianosman@google.com> Bug: skia:10396 Change-Id: If1347bcacb7c405a66f9d4c5b0059e9d735b3f9a Reviewed-on: https://skia-review.googlesource.com/c/skia/+/298062 Reviewed-by: Michael Ludwig <michaelludwig@google.com> Commit-Queue: Michael Ludwig <michaelludwig@google.com>
2020-06-19 14:47:30 +00:00
}
void addToKey(const GrShaderCaps& shaderCaps, skgpu::KeyBuilder* b) const override {
b->add32(ProgramImpl::ComputeMatrixKey(shaderCaps, fLocalMatrix));
Reland "Centralize geometry processor vertex shader transform code" This is a reland of 0426947243e1d6b470830c4d3b1c5704ed1f23e1 Original change's description: > Centralize geometry processor vertex shader transform code > > GrGLSLGeometryProcessors no longer have to call emitTransforms() in > their onEmitCode() function. Instead, the GpArgs struct allows them to > set a GrShaderVar that holds the computed or explicitly provided local > coordinates in the vertex shader. > > The base GrGLSLGeometryProcessor now automatically uses that to collect > all of the transforms that can then be lifted out of FPs to the vertex > shader, and base their computation on the GP provided local coordinate. > > As part of this, there is no more built-in magic concatenation of a > local matrix / inverse view matrix to these coordinate transforms. GP > implementations that relied on this now manage their own uniform for this > matrix and compute the local coordinate before assigning to GpArgs. > > The base GrGLSLGeometryProcessor is updated to provide helpers for this > pattern. > > Bug: skia:10396 > Change-Id: I56afb3fff4b806f6015ab13626ac1afde9ef5c2b > Reviewed-on: https://skia-review.googlesource.com/c/skia/+/297027 > Commit-Queue: Michael Ludwig <michaelludwig@google.com> > Reviewed-by: Brian Osman <brianosman@google.com> Bug: skia:10396 Change-Id: If1347bcacb7c405a66f9d4c5b0059e9d735b3f9a Reviewed-on: https://skia-review.googlesource.com/c/skia/+/298062 Reviewed-by: Michael Ludwig <michaelludwig@google.com> Commit-Queue: Michael Ludwig <michaelludwig@google.com>
2020-06-19 14:47:30 +00:00
}
bool wideColor() const { return fInColor.cpuType() != kUByte4_norm_GrVertexAttribType; }
private:
Attribute fInPosition = { "inPosition", kFloat2_GrVertexAttribType, kFloat2_GrSLType};
Reland "Centralize geometry processor vertex shader transform code" This is a reland of 0426947243e1d6b470830c4d3b1c5704ed1f23e1 Original change's description: > Centralize geometry processor vertex shader transform code > > GrGLSLGeometryProcessors no longer have to call emitTransforms() in > their onEmitCode() function. Instead, the GpArgs struct allows them to > set a GrShaderVar that holds the computed or explicitly provided local > coordinates in the vertex shader. > > The base GrGLSLGeometryProcessor now automatically uses that to collect > all of the transforms that can then be lifted out of FPs to the vertex > shader, and base their computation on the GP provided local coordinate. > > As part of this, there is no more built-in magic concatenation of a > local matrix / inverse view matrix to these coordinate transforms. GP > implementations that relied on this now manage their own uniform for this > matrix and compute the local coordinate before assigning to GpArgs. > > The base GrGLSLGeometryProcessor is updated to provide helpers for this > pattern. > > Bug: skia:10396 > Change-Id: I56afb3fff4b806f6015ab13626ac1afde9ef5c2b > Reviewed-on: https://skia-review.googlesource.com/c/skia/+/297027 > Commit-Queue: Michael Ludwig <michaelludwig@google.com> > Reviewed-by: Brian Osman <brianosman@google.com> Bug: skia:10396 Change-Id: If1347bcacb7c405a66f9d4c5b0059e9d735b3f9a Reviewed-on: https://skia-review.googlesource.com/c/skia/+/298062 Reviewed-by: Michael Ludwig <michaelludwig@google.com> Commit-Queue: Michael Ludwig <michaelludwig@google.com>
2020-06-19 14:47:30 +00:00
Attribute fInLocalCoords = {"inLocalCoords", kFloat2_GrVertexAttribType, kFloat2_GrSLType};
Attribute fInColor;
Reland "Centralize geometry processor vertex shader transform code" This is a reland of 0426947243e1d6b470830c4d3b1c5704ed1f23e1 Original change's description: > Centralize geometry processor vertex shader transform code > > GrGLSLGeometryProcessors no longer have to call emitTransforms() in > their onEmitCode() function. Instead, the GpArgs struct allows them to > set a GrShaderVar that holds the computed or explicitly provided local > coordinates in the vertex shader. > > The base GrGLSLGeometryProcessor now automatically uses that to collect > all of the transforms that can then be lifted out of FPs to the vertex > shader, and base their computation on the GP provided local coordinate. > > As part of this, there is no more built-in magic concatenation of a > local matrix / inverse view matrix to these coordinate transforms. GP > implementations that relied on this now manage their own uniform for this > matrix and compute the local coordinate before assigning to GpArgs. > > The base GrGLSLGeometryProcessor is updated to provide helpers for this > pattern. > > Bug: skia:10396 > Change-Id: I56afb3fff4b806f6015ab13626ac1afde9ef5c2b > Reviewed-on: https://skia-review.googlesource.com/c/skia/+/297027 > Commit-Queue: Michael Ludwig <michaelludwig@google.com> > Reviewed-by: Brian Osman <brianosman@google.com> Bug: skia:10396 Change-Id: If1347bcacb7c405a66f9d4c5b0059e9d735b3f9a Reviewed-on: https://skia-review.googlesource.com/c/skia/+/298062 Reviewed-by: Michael Ludwig <michaelludwig@google.com> Commit-Queue: Michael Ludwig <michaelludwig@google.com>
2020-06-19 14:47:30 +00:00
SkMatrix fLocalMatrix;
};
class TestRectOp final : public GrMeshDrawOp {
public:
static GrOp::Owner Make(GrRecordingContext*,
GrPaint&&,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localM);
const char* name() const override { return "TestRectOp"; }
FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; }
GrProcessorSet::Analysis finalize(const GrCaps&,
const GrAppliedClip*,
GrClampType) override;
void visitProxies(const GrVisitProxyFunc& func) const override {
if (fProgramInfo) {
fProgramInfo->visitFPProxies(func);
} else {
fProcessorSet.visitProxies(func);
}
}
private:
DEFINE_OP_CLASS_ID
TestRectOp(const GrCaps*,
GrPaint&&,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localMatrix);
GrProgramInfo* programInfo() override { return fProgramInfo; }
void onCreateProgramInfo(const GrCaps*,
SkArenaAlloc*,
const GrSurfaceProxyView& writeView,
bool usesMSAASurface,
GrAppliedClip&&,
const GrDstProxyView&,
GrXferBarrierFlags renderPassXferBarriers,
GrLoadOp colorLoadOp) override;
void onPrepareDraws(GrMeshDrawTarget*) override;
void onExecute(GrOpFlushState*, const SkRect& chainBounds) override;
SkRect fDrawRect;
SkRect fLocalRect;
SkPMColor4f fColor;
GP fGP;
GrProcessorSet fProcessorSet;
// If this op is prePrepared the created programInfo will be stored here for use in
// onExecute. In the prePrepared case it will have been stored in the record-time arena.
GrProgramInfo* fProgramInfo = nullptr;
GrSimpleMesh* fMesh = nullptr;
friend class ::GrOp;
};
GrOp::Owner TestRectOp::Make(GrRecordingContext* context,
GrPaint&& paint,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localM) {
const auto* caps = context->priv().caps();
return GrOp::Make<TestRectOp>(context, caps, std::move(paint), drawRect, localRect, localM);
}
GrProcessorSet::Analysis TestRectOp::finalize(const GrCaps& caps,
const GrAppliedClip* clip,
GrClampType clampType) {
return fProcessorSet.finalize(GrProcessorAnalysisColor::Opaque::kYes,
GrProcessorAnalysisCoverage::kSingleChannel, clip,
&GrUserStencilSettings::kUnused, caps, clampType, &fColor);
}
static bool use_wide_color(const GrPaint& paint, const GrCaps* caps) {
return !paint.getColor4f().fitsInBytes() && caps->halfFloatVertexAttributeSupport();
}
TestRectOp::TestRectOp(const GrCaps* caps,
GrPaint&& paint,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localMatrix)
: GrMeshDrawOp(ClassID())
, fDrawRect(drawRect)
, fLocalRect(localRect)
, fColor(paint.getColor4f())
, fGP(localMatrix, use_wide_color(paint, caps))
, fProcessorSet(std::move(paint)) {
this->setBounds(drawRect.makeSorted(), HasAABloat::kNo, IsHairline::kNo);
}
void TestRectOp::onCreateProgramInfo(const GrCaps* caps,
SkArenaAlloc* arena,
const GrSurfaceProxyView& writeView,
bool usesMSAASurface,
GrAppliedClip&& appliedClip,
const GrDstProxyView& dstProxyView,
GrXferBarrierFlags renderPassXferBarriers,
GrLoadOp colorLoadOp) {
fProgramInfo = GrSimpleMeshDrawOpHelper::CreateProgramInfo(caps,
arena,
writeView,
usesMSAASurface,
std::move(appliedClip),
dstProxyView,
&fGP,
std::move(fProcessorSet),
GrPrimitiveType::kTriangles,
renderPassXferBarriers,
colorLoadOp,
GrPipeline::InputFlags::kNone);
}
void TestRectOp::onPrepareDraws(GrMeshDrawTarget* target) {
QuadHelper helper(target, fGP.vertexStride(), 1);
skgpu::VertexWriter writer{helper.vertices()};
auto pos = skgpu::VertexWriter::TriStripFromRect(fDrawRect);
auto local = skgpu::VertexWriter::TriStripFromRect(fLocalRect);
skgpu::VertexColor color(fColor, fGP.wideColor());
writer.writeQuad(pos, local, color);
fMesh = helper.mesh();
}
void TestRectOp::onExecute(GrOpFlushState* flushState, const SkRect& chainBounds) {
if (!fProgramInfo) {
this->createProgramInfo(flushState);
}
flushState->bindPipelineAndScissorClip(*fProgramInfo, chainBounds);
flushState->bindTextures(fProgramInfo->geomProc(), nullptr, fProgramInfo->pipeline());
flushState->drawMesh(*fMesh);
}
} // anonymous namespace
namespace sk_gpu_test::test_ops {
GrOp::Owner MakeRect(GrRecordingContext* context,
GrPaint&& paint,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localM) {
return TestRectOp::Make(context, std::move(paint), drawRect, localRect, localM);
}
GrOp::Owner MakeRect(GrRecordingContext* context,
std::unique_ptr<GrFragmentProcessor> fp,
const SkRect& drawRect,
const SkRect& localRect,
const SkMatrix& localM) {
GrPaint paint;
paint.setColorFragmentProcessor(std::move(fp));
return TestRectOp::Make(context, std::move(paint), drawRect, localRect, localM);
}
GrOp::Owner MakeRect(GrRecordingContext* context,
GrPaint&& paint,
const SkRect& rect) {
return TestRectOp::Make(context, std::move(paint), rect, rect, SkMatrix::I());
}
} // namespace sk_gpu_test::test_ops