blob: 10460d65cc47732931ae36a3a0d947eea06f49dc [file] [log] [blame]
/*
* Copyright 2022 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkBlendMode.h"
#include "include/core/SkColor.h"
#include "include/core/SkColorSpace.h"
#include "include/core/SkPoint.h"
#include "include/core/SkRect.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkSurfaceProps.h"
#include "include/core/SkTypes.h"
#include "include/gpu/ganesh/GrDirectContext.h"
#include "include/private/SkColorData.h"
#include "include/private/base/SkAlign.h"
#include "include/private/base/SkTemplates.h"
#include "include/private/gpu/ganesh/GrTypesPriv.h"
#include "src/core/SkSLTypeShared.h"
#include "src/gpu/SkBackingFit.h"
#include "src/gpu/ganesh/GrAppliedClip.h"
#include "src/gpu/ganesh/GrBuffer.h"
#include "src/gpu/ganesh/GrCaps.h"
#include "src/gpu/ganesh/GrDirectContextPriv.h"
#include "src/gpu/ganesh/GrDrawingManager.h"
#include "src/gpu/ganesh/GrGeometryProcessor.h"
#include "src/gpu/ganesh/GrGpu.h"
#include "src/gpu/ganesh/GrGpuBuffer.h"
#include "src/gpu/ganesh/GrImageInfo.h"
#include "src/gpu/ganesh/GrMeshDrawTarget.h"
#include "src/gpu/ganesh/GrOpFlushState.h"
#include "src/gpu/ganesh/GrPipeline.h"
#include "src/gpu/ganesh/GrPixmap.h"
#include "src/gpu/ganesh/GrProcessorAnalysis.h"
#include "src/gpu/ganesh/GrProcessorSet.h"
#include "src/gpu/ganesh/GrProgramInfo.h"
#include "src/gpu/ganesh/GrResourceProvider.h"
#include "src/gpu/ganesh/GrSimpleMesh.h"
#include "src/gpu/ganesh/GrUserStencilSettings.h"
#include "src/gpu/ganesh/SurfaceDrawContext.h"
#include "src/gpu/ganesh/glsl/GrGLSLFragmentShaderBuilder.h"
#include "src/gpu/ganesh/glsl/GrGLSLVarying.h"
#include "src/gpu/ganesh/ops/GrMeshDrawOp.h"
#include "src/gpu/ganesh/ops/GrOp.h"
#include "src/gpu/ganesh/ops/GrSimpleMeshDrawOpHelper.h"
#include "tests/CtsEnforcement.h"
#include "tests/Test.h"
#include <algorithm>
#include <cstring>
#include <initializer_list>
#include <memory>
#include <string_view>
#include <utility>
class GrDstProxyView;
class GrGLSLProgramDataManager;
class GrRecordingContext;
class GrSurfaceProxyView;
class SkArenaAlloc;
enum class GrXferBarrierFlags;
namespace skgpu { class KeyBuilder; }
struct GrContextOptions;
struct GrShaderCaps;
// Simple op that draws a vertex buffer with float2 positions as green triangles. We use this to
// draw GrGpuBuffers to test that the buffer contains the expected values as not all contexts will
// support buffer mapping.
class TestVertexOp final : public GrMeshDrawOp {
public:
static GrOp::Owner Make(GrRecordingContext* context,
sk_sp<GrGpuBuffer> buffer,
int baseVertex,
int vertexCount,
const SkRect& bounds) {
return GrOp::Make<TestVertexOp>(context,
std::move(buffer),
baseVertex,
vertexCount,
bounds);
}
const char* name() const override { return "TestVertexOp"; }
FixedFunctionFlags fixedFunctionFlags() const override { return FixedFunctionFlags::kNone; }
GrProcessorSet::Analysis finalize(const GrCaps& caps,
const GrAppliedClip* clip,
GrClampType clampType) override {
static constexpr SkPMColor4f kGreen{0, 1, 0, 1};
SkPMColor4f color = kGreen;
auto analysis = fProcessorSet.finalize(GrProcessorAnalysisColor::Opaque::kYes,
GrProcessorAnalysisCoverage::kNone,
clip,
&GrUserStencilSettings::kUnused,
caps,
clampType,
&color);
SkASSERT(color == kGreen);
return analysis;
}
void visitProxies(const GrVisitProxyFunc& func) const override {
if (fProgramInfo) {
fProgramInfo->visitFPProxies(func);
}
}
private:
DEFINE_OP_CLASS_ID
TestVertexOp(sk_sp<GrGpuBuffer> buffer,
int baseVertex,
int vertexCount,
const SkRect& bounds)
: GrMeshDrawOp(ClassID())
, fBuffer(std::move(buffer))
, fProcessorSet(SkBlendMode::kSrc)
, fBaseVertex(baseVertex)
, fVertexCount(vertexCount) {
this->setBounds(bounds, HasAABloat::kNo, GrOp::IsHairline::kNo);
}
GrProgramInfo* programInfo() override { return fProgramInfo; }
void onCreateProgramInfo(const GrCaps* caps,
SkArenaAlloc* arena,
const GrSurfaceProxyView& writeView,
bool usesMSAASurface,
GrAppliedClip&& appliedClip,
const GrDstProxyView& dstProxyView,
GrXferBarrierFlags renderPassXferBarriers,
GrLoadOp colorLoadOp) override {
fProgramInfo = GrSimpleMeshDrawOpHelper::CreateProgramInfo(
caps,
arena,
writeView,
usesMSAASurface,
std::move(appliedClip),
dstProxyView,
&fGP,
std::move(fProcessorSet),
GrPrimitiveType::kTriangles,
renderPassXferBarriers,
colorLoadOp,
GrPipeline::InputFlags::kNone);
}
class GP : public GrGeometryProcessor {
public:
GP() : GrGeometryProcessor(kTestFP_ClassID) {
this->setVertexAttributesWithImplicitOffsets(&kPos, 1);
}
const char* name() const override { return "TestVertexOp::GP"; }
std::unique_ptr<ProgramImpl> makeProgramImpl(const GrShaderCaps&) const override {
class Impl : public ProgramImpl {
public:
void setData(const GrGLSLProgramDataManager&,
const GrShaderCaps&,
const GrGeometryProcessor&) override {}
private:
void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
const auto& gp = args.fGeomProc.cast<GP>();
args.fVaryingHandler->emitAttributes(gp);
args.fFragBuilder->codeAppendf("half4 %s = half4(0, 1, 0, 1);",
args.fOutputColor);
args.fFragBuilder->codeAppendf("const half4 %s = half4(1);",
args.fOutputCoverage);
WriteOutputPosition(args.fVertBuilder, gpArgs, kPos.name());
}
UniformHandle fLocalMatrixUni;
};
return std::make_unique<Impl>();
}
void addToKey(const GrShaderCaps &caps, skgpu::KeyBuilder *builder) const override {}
private:
static constexpr Attribute kPos = {"pos", kFloat2_GrVertexAttribType, SkSLType::kFloat2};
};
void onPrepareDraws(GrMeshDrawTarget* target) override {
fMesh = target->allocMesh();
fMesh->set(fBuffer, fVertexCount, fBaseVertex);
}
void onExecute(GrOpFlushState* flushState, const SkRect& chainBounds) override {
if (!fProgramInfo) {
this->createProgramInfo(flushState);
}
flushState->bindPipelineAndScissorClip(*fProgramInfo, chainBounds);
flushState->bindTextures(fProgramInfo->geomProc(), nullptr, fProgramInfo->pipeline());
flushState->drawMesh(*fMesh);
}
sk_sp<GrGpuBuffer> fBuffer;
GP fGP;
GrProcessorSet fProcessorSet;
int fBaseVertex;
int fVertexCount;
GrProgramInfo* fProgramInfo = nullptr;
GrSimpleMesh* fMesh = nullptr;
friend class ::GrOp;
};
DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(GrGpuBufferTransferTest,
reporter,
ctxInfo,
CtsEnforcement::kApiLevel_U) {
if (!ctxInfo.directContext()->priv().caps()->transferFromBufferToBufferSupport()) {
return;
}
GrDirectContext* dc = ctxInfo.directContext();
GrDrawingManager* dm = dc->priv().drawingManager();
GrResourceProvider* rp = ctxInfo.directContext()->priv().resourceProvider();
GrGpu* gpu = ctxInfo.directContext()->priv().getGpu();
auto create_cpu_to_gpu_buffer = [&](int baseVertex) {
// Ensure any extra vertices are offscreen
int totalVertices = baseVertex + 6;
auto points = std::make_unique<SkPoint[]>(totalVertices);
SkPoint offscreenPt{-10000, -10000};
std::fill_n(points.get(), totalVertices, offscreenPt);
// set the quad at the desired base vertex
static constexpr SkPoint kUnitQuad[] {{0, 0}, {0, 1}, {1, 0},
{1, 0}, {0, 1}, {1, 1}};
std::copy_n(kUnitQuad, 6, points.get() + baseVertex);
return rp->createBuffer(points.get(),
totalVertices*sizeof(SkPoint),
GrGpuBufferType::kXferCpuToGpu,
kDynamic_GrAccessPattern);
};
auto create_vertex_buffer = [&](sk_sp<GrGpuBuffer> srcBuffer,
int srcBaseVertex,
int vbBaseVertex,
bool useTask,
bool minSizedTransfers) {
// make initialization data of offscreen points.
int dstVertexCount = vbBaseVertex + 6;
auto points = std::make_unique<SkPoint[]>(dstVertexCount);
SkPoint offscreenPt{-10000, -10000};
std::fill_n(points.get(), dstVertexCount, offscreenPt);
sk_sp<GrGpuBuffer> vb = rp->createBuffer(points.get(),
dstVertexCount*sizeof(SkPoint),
GrGpuBufferType::kVertex,
kDynamic_GrAccessPattern);
// copy actual quad data from the source buffer to our new vb.
static constexpr size_t kTotalSize = 6*sizeof(SkPoint);
size_t srcOffset = srcBaseVertex*sizeof(SkPoint);
size_t vbOffset = vbBaseVertex*sizeof(SkPoint);
size_t alignment = gpu->caps()->transferFromBufferToBufferAlignment();
SkASSERT(kTotalSize % alignment == 0);
SkASSERT(sizeof(SkPoint) % alignment == 0);
if (minSizedTransfers) {
for (size_t n = kTotalSize/alignment, i = 0; i < n; ++i) {
if (useTask) {
dm->newBufferTransferTask(srcBuffer,
srcOffset + i*alignment,
vb,
vbOffset + i*alignment,
alignment);
} else {
gpu->transferFromBufferToBuffer(srcBuffer,
srcOffset + i*alignment,
vb,
vbOffset + i*alignment,
alignment);
}
}
} else if (useTask) {
dm->newBufferTransferTask(srcBuffer, srcOffset, vb, vbOffset, kTotalSize);
} else {
gpu->transferFromBufferToBuffer(srcBuffer, srcOffset, vb, vbOffset, kTotalSize);
}
return vb;
};
auto sdc = skgpu::ganesh::SurfaceDrawContext::Make(dc,
GrColorType::kRGBA_8888,
nullptr,
SkBackingFit::kExact,
{1, 1},
SkSurfaceProps{},
std::string_view{});
if (!sdc) {
ERRORF(reporter, "Could not create draw context");
return;
}
auto pm = GrPixmap::Allocate(sdc->imageInfo().makeColorType(GrColorType::kRGBA_F32));
for (bool useTask : {false, true}) {
for (bool minSizedTransfers : {false, true}) {
for (int srcBaseVertex : {0, 5}) {
auto src = create_cpu_to_gpu_buffer(srcBaseVertex);
if (!src) {
ERRORF(reporter, "Could not create src buffer");
return;
}
for (int vbBaseVertex : {0, 2}) {
auto vb = create_vertex_buffer(src,
srcBaseVertex,
vbBaseVertex,
useTask,
minSizedTransfers);
if (!vb) {
ERRORF(reporter, "Could not create vertex buffer");
return;
}
static constexpr SkColor4f kRed{1, 0, 0, 1};
static constexpr SkRect kBounds{0, 0, 1, 1};
sdc->clear(kRed);
sdc->addDrawOp(nullptr, TestVertexOp::Make(dc,
vb,
vbBaseVertex,
/*vertexCount=*/6,
kBounds));
auto color = static_cast<SkPMColor4f*>(pm.addr());
*color = kRed.premul();
if (!sdc->readPixels(dc, pm, {0, 0})) {
ERRORF(reporter, "Read back failed.");
return;
}
static constexpr SkPMColor4f kGreen{0, 1, 0, 1};
REPORTER_ASSERT(reporter, *color == kGreen, "src base vertex: %d, "
"vb base vertex: %d, "
"use task: %d, "
"minSizedTransfers: %d",
srcBaseVertex,
vbBaseVertex,
useTask,
minSizedTransfers);
}
}
}
}
}
DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(GrGpuBufferUpdateDataTest,
reporter,
ctxInfo,
CtsEnforcement::kApiLevel_U) {
GrDirectContext* dc = ctxInfo.directContext();
GrGpu* gpu = ctxInfo.directContext()->priv().getGpu();
static constexpr SkPoint kUnitQuad[] {{0, 0}, {0, 1}, {1, 0},
{1, 0}, {0, 1}, {1, 1}};
auto sdc = skgpu::ganesh::SurfaceDrawContext::Make(dc,
GrColorType::kRGBA_8888,
nullptr,
SkBackingFit::kExact,
{1, 1},
SkSurfaceProps{},
std::string_view{});
if (!sdc) {
ERRORF(reporter, "Could not create draw context");
return;
}
auto pm = GrPixmap::Allocate(sdc->imageInfo().makeColorType(GrColorType::kRGBA_F32));
for (bool piecewise : {false, true}) {
size_t alignment = piecewise ? gpu->caps()->bufferUpdateDataPreserveAlignment() : 1;
for (size_t offset : {size_t{0}, 4*sizeof(SkPoint), size_t{1}, size_t{27}}) {
// For non-discarding updates we may not be able to actually put the data at an
// arbitrary offset.
if (alignment > 1) {
offset = SkAlignTo(offset, alignment);
}
for (auto accessPattern : {kStatic_GrAccessPattern,
// kStream_GrAccessPattern, GrVkGpu asserts on this for VBs.
kDynamic_GrAccessPattern}) {
// Go direct to GrGpu to avoid caching/size adjustments at GrResourceProvider level.
// We add an extra size(SkPoint) to ensure that everything fits when we align the
// first point's location in the vb below.
auto vb = gpu->createBuffer(sizeof(kUnitQuad) + offset + sizeof(SkPoint),
GrGpuBufferType::kVertex,
accessPattern);
if (!vb) {
ERRORF(reporter, "Could not create vertex buffer");
return;
}
const void* src = kUnitQuad;
size_t updateSize = sizeof(kUnitQuad);
// The vertices in the VB must be aligned to the size of a vertex (because our draw
// call takes a base vertex index rather than a byte offset). So if we want our
// upload to begin at a non-aligned byte we shift the data in the src buffer so that
// it falls at a vertex alignment in the vb.
std::unique_ptr<char[]> tempSrc;
size_t baseVertex = offset/sizeof(SkPoint);
if (size_t r = offset%sizeof(SkPoint); r != 0) {
size_t pad = sizeof(SkPoint) - r;
updateSize += pad;
if (alignment > 1) {
updateSize = SkAlignTo(updateSize, alignment);
}
++baseVertex;
tempSrc.reset(new char[updateSize]);
std::memcpy(tempSrc.get() + pad, kUnitQuad, sizeof(kUnitQuad));
src = tempSrc.get();
}
if (piecewise) {
// This is the minimum size we can transfer at once.
size_t pieceSize = alignment;
// Upload each piece from a buffer where the byte before and after the uploaded
// bytes are not the same values as want adjacent to the piece in the buffer.
// Thus, if updateData() transfers extra bytes around the source we should get a
// bad buffer.
auto piece = std::make_unique<unsigned char[]>(pieceSize + 2);
piece[0] = piece[pieceSize + 1] = 0xFF;
for (size_t o = 0; o < updateSize; o += pieceSize) {
memcpy(&piece[1], SkTAddOffset<const void>(src, o), pieceSize);
if (!vb->updateData(&piece[1], offset + o, pieceSize, /*preserve=*/true)) {
ERRORF(reporter, "GrGpuBuffer::updateData returned false.");
return;
}
}
} else if (!vb->updateData(src, offset, updateSize, /*preserve=*/false)) {
ERRORF(reporter, "GrGpuBuffer::updateData returned false.");
return;
}
static constexpr SkColor4f kRed{1, 0, 0, 1};
static constexpr SkRect kBounds{0, 0, 1, 1};
sdc->clear(kRed);
sdc->addDrawOp(nullptr, TestVertexOp::Make(dc,
vb,
baseVertex,
std::size(kUnitQuad),
kBounds));
auto color = static_cast<SkPMColor4f*>(pm.addr());
*color = kRed.premul();
if (!sdc->readPixels(dc, pm, {0, 0})) {
ERRORF(reporter, "Read back failed.");
return;
}
static constexpr SkPMColor4f kGreen{0, 1, 0, 1};
REPORTER_ASSERT(reporter, *color == kGreen, "piecewise: %d, offset: %zu",
piecewise, offset);
}
}
}
}