blob: 0ce2eed06876cb59cd51d5a0d728f2b5225e40f1 [file] [log] [blame]
/*
* Copyright 2014 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/core/SkMatrixProvider.h"
#include "src/core/SkTLazy.h"
#include "src/core/SkVM.h"
#include "src/shaders/SkLocalMatrixShader.h"
#if SK_SUPPORT_GPU
#include "src/gpu/ganesh/GrFPArgs.h"
#include "src/gpu/ganesh/GrFragmentProcessor.h"
#include "src/gpu/ganesh/effects/GrMatrixEffect.h"
#endif
#ifdef SK_GRAPHITE_ENABLED
#include "src/gpu/graphite/KeyContext.h"
#include "src/gpu/graphite/KeyHelpers.h"
#include "src/gpu/graphite/PaintParamsKey.h"
#endif
SkShaderBase::GradientType SkLocalMatrixShader::asGradient(GradientInfo* info,
SkMatrix* localMatrix) const {
GradientType type = as_SB(fWrappedShader)->asGradient(info, localMatrix);
if (type != SkShaderBase::GradientType::kNone && localMatrix) {
*localMatrix = ConcatLocalMatrices(fLocalMatrix, *localMatrix);
}
return type;
}
#if SK_SUPPORT_GPU
std::unique_ptr<GrFragmentProcessor> SkLocalMatrixShader::asFragmentProcessor(
const GrFPArgs& args) const {
return as_SB(fWrappedShader)->asFragmentProcessor(GrFPArgs::ConcatLocalMatrix(args,
fLocalMatrix));
}
#endif
#ifdef SK_GRAPHITE_ENABLED
void SkLocalMatrixShader::addToKey(const skgpu::graphite::KeyContext& keyContext,
skgpu::graphite::PaintParamsKeyBuilder* builder,
skgpu::graphite::PipelineDataGatherer* gatherer) const {
using namespace skgpu::graphite;
LocalMatrixShaderBlock::LMShaderData lmShaderData(fLocalMatrix);
KeyContextWithLocalMatrix newContext(keyContext, fLocalMatrix);
LocalMatrixShaderBlock::BeginBlock(newContext, builder, gatherer, &lmShaderData);
as_SB(fWrappedShader)->addToKey(newContext, builder, gatherer);
builder->endBlock();
}
#endif
sk_sp<SkFlattenable> SkLocalMatrixShader::CreateProc(SkReadBuffer& buffer) {
SkMatrix lm;
buffer.readMatrix(&lm);
auto baseShader(buffer.readShader());
if (!baseShader) {
return nullptr;
}
return baseShader->makeWithLocalMatrix(lm);
}
void SkLocalMatrixShader::flatten(SkWriteBuffer& buffer) const {
buffer.writeMatrix(fLocalMatrix);
buffer.writeFlattenable(fWrappedShader.get());
}
#ifdef SK_ENABLE_LEGACY_SHADERCONTEXT
SkShaderBase::Context* SkLocalMatrixShader::onMakeContext(
const ContextRec& rec, SkArenaAlloc* alloc) const
{
SkTCopyOnFirstWrite<SkMatrix> lm(fLocalMatrix);
if (rec.fLocalMatrix) {
*lm.writable() = ConcatLocalMatrices(*rec.fLocalMatrix, *lm);
}
ContextRec newRec(rec);
newRec.fLocalMatrix = lm;
return as_SB(fWrappedShader)->makeContext(newRec, alloc);
}
#endif
SkImage* SkLocalMatrixShader::onIsAImage(SkMatrix* outMatrix, SkTileMode* mode) const {
SkMatrix imageMatrix;
SkImage* image = fWrappedShader->isAImage(&imageMatrix, mode);
if (image && outMatrix) {
*outMatrix = ConcatLocalMatrices(fLocalMatrix, imageMatrix);
}
return image;
}
bool SkLocalMatrixShader::onAppendStages(const SkStageRec& rec) const {
SkTCopyOnFirstWrite<SkMatrix> lm(fLocalMatrix);
if (rec.fLocalM) {
*lm.writable() = ConcatLocalMatrices(*rec.fLocalM, *lm);
}
SkStageRec newRec = rec;
newRec.fLocalM = lm;
return as_SB(fWrappedShader)->appendStages(newRec);
}
skvm::Color SkLocalMatrixShader::onProgram(skvm::Builder* p,
skvm::Coord device, skvm::Coord local, skvm::Color paint,
const SkMatrixProvider& matrices, const SkMatrix* localM,
const SkColorInfo& dst,
skvm::Uniforms* uniforms, SkArenaAlloc* alloc) const {
SkTCopyOnFirstWrite<SkMatrix> lm(fLocalMatrix);
if (localM) {
*lm.writable() = ConcatLocalMatrices(*localM, *lm);
}
return as_SB(fWrappedShader)->program(p, device,local, paint,
matrices,lm.get(), dst,
uniforms,alloc);
}
sk_sp<SkShader> SkShader::makeWithLocalMatrix(const SkMatrix& localMatrix) const {
if (localMatrix.isIdentity()) {
return sk_ref_sp(const_cast<SkShader*>(this));
}
const SkMatrix* lm = &localMatrix;
sk_sp<SkShader> baseShader;
SkMatrix otherLocalMatrix;
sk_sp<SkShader> proxy = as_SB(this)->makeAsALocalMatrixShader(&otherLocalMatrix);
if (proxy) {
otherLocalMatrix = SkShaderBase::ConcatLocalMatrices(localMatrix, otherLocalMatrix);
lm = &otherLocalMatrix;
baseShader = proxy;
} else {
baseShader = sk_ref_sp(const_cast<SkShader*>(this));
}
return sk_make_sp<SkLocalMatrixShader>(std::move(baseShader), *lm);
}
////////////////////////////////////////////////////////////////////
/**
* Replaces the CTM when used. Created to support clipShaders, which have to be evaluated
* using the CTM that was present at the time they were specified (which may be different
* from the CTM at the time something is drawn through the clip.
*/
class SkCTMShader final : public SkShaderBase {
public:
SkCTMShader(sk_sp<SkShader> proxy, const SkMatrix& ctm)
: fProxyShader(std::move(proxy))
, fCTM(ctm)
{}
GradientType asGradient(GradientInfo* info, SkMatrix* localMatrix) const override {
return as_SB(fProxyShader)->asGradient(info, localMatrix);
}
#if SK_SUPPORT_GPU
std::unique_ptr<GrFragmentProcessor> asFragmentProcessor(const GrFPArgs&) const override;
#endif
protected:
void flatten(SkWriteBuffer&) const override { SkASSERT(false); }
bool onAppendStages(const SkStageRec& rec) const override {
SkOverrideDeviceMatrixProvider matrixProvider(fCTM);
SkStageRec newRec = {
rec.fPipeline,
rec.fAlloc,
rec.fDstColorType,
rec.fDstCS,
rec.fPaint,
rec.fLocalM,
matrixProvider,
rec.fSurfaceProps
};
return as_SB(fProxyShader)->appendStages(newRec);
}
skvm::Color onProgram(skvm::Builder* p,
skvm::Coord device, skvm::Coord local, skvm::Color paint,
const SkMatrixProvider& matrices, const SkMatrix* localM,
const SkColorInfo& dst,
skvm::Uniforms* uniforms, SkArenaAlloc* alloc) const override {
SkOverrideDeviceMatrixProvider matrixProvider(fCTM);
return as_SB(fProxyShader)->program(p, device,local, paint,
matrixProvider,localM, dst,
uniforms,alloc);
}
private:
SK_FLATTENABLE_HOOKS(SkCTMShader)
sk_sp<SkShader> fProxyShader;
SkMatrix fCTM;
using INHERITED = SkShaderBase;
};
#if SK_SUPPORT_GPU
std::unique_ptr<GrFragmentProcessor> SkCTMShader::asFragmentProcessor(
const GrFPArgs& args) const {
SkMatrix ctmInv;
if (!fCTM.invert(&ctmInv)) {
return nullptr;
}
auto ctmProvider = SkOverrideDeviceMatrixProvider(fCTM);
auto base = as_SB(fProxyShader)->asFragmentProcessor(args.withNewMatrixProvider(ctmProvider));
if (!base) {
return nullptr;
}
// In order for the shader to be evaluated with the original CTM, we explicitly evaluate it
// at sk_FragCoord, and pass that through the inverse of the original CTM. This avoids requiring
// local coords for the shader and mapping from the draw's local to device and then back.
return GrFragmentProcessor::DeviceSpace(GrMatrixEffect::Make(ctmInv, std::move(base)));
}
#endif
sk_sp<SkFlattenable> SkCTMShader::CreateProc(SkReadBuffer& buffer) {
SkASSERT(false);
return nullptr;
}
sk_sp<SkShader> SkShaderBase::makeWithCTM(const SkMatrix& postM) const {
return sk_sp<SkShader>(new SkCTMShader(sk_ref_sp(this), postM));
}