/* * Copyright 2019 Google LLC * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef SkRuntimeEffect_DEFINED #define SkRuntimeEffect_DEFINED #include "include/core/SkString.h" #include #if SK_SUPPORT_GPU #include "include/private/GrTypesPriv.h" #endif class GrShaderCaps; class SkColorFilter; class SkMatrix; class SkShader; namespace SkSL { class ByteCode; struct PipelineStageArgs; struct Program; class SharedCompiler; } /* * SkRuntimeEffect supports creating custom SkShader and SkColorFilter objects using Skia's SkSL * shading language. * * * This API is experimental and subject to change. */ class SK_API SkRuntimeEffect : public SkRefCnt { public: struct Variable { enum class Qualifier { kUniform, kIn, }; enum class Type { kBool, kInt, kFloat, kFloat2, kFloat3, kFloat4, kFloat2x2, kFloat3x3, kFloat4x4, }; enum Flags { kArray_Flag = 0x1, }; SkString fName; size_t fOffset; Qualifier fQualifier; Type fType; int fCount; uint32_t fFlags; #if SK_SUPPORT_GPU GrSLType fGPUType; #endif bool isArray() const { return SkToBool(fFlags & kArray_Flag); } size_t sizeInBytes() const; }; // [Effect, ErrorText] // If successful, Effect != nullptr, otherwise, ErrorText contains the reason for failure. using EffectResult = std::tuple, SkString>; static EffectResult Make(SkString sksl); sk_sp makeShader(sk_sp inputs, sk_sp children[], size_t childCount, const SkMatrix* localMatrix, bool isOpaque); sk_sp makeColorFilter(sk_sp inputs, sk_sp children[], size_t childCount); sk_sp makeColorFilter(sk_sp inputs); const SkString& source() const { return fSkSL; } uint32_t hash() const { return fHash; } template class ConstIterable { public: ConstIterable(const std::vector& vec) : fVec(vec) {} using const_iterator = typename std::vector::const_iterator; const_iterator begin() const { return fVec.begin(); } const_iterator end() const { return fVec.end(); } size_t count() const { return fVec.size(); } private: const std::vector& fVec; }; // Combined size of all 'in' and 'uniform' variables. When calling makeColorFilter or // makeShader, provide an SkData of this size, containing values for all of those variables. size_t inputSize() const; // Combined size of just the 'uniform' variables. size_t uniformSize() const { return fUniformSize; } ConstIterable inputs() const { return ConstIterable(fInAndUniformVars); } ConstIterable children() const { return ConstIterable(fChildren); } #if SK_SUPPORT_GPU // This re-compiles the program from scratch, using the supplied shader caps. // This is necessary to get the correct values of settings. bool toPipelineStage(const void* inputs, const GrShaderCaps* shaderCaps, SkSL::PipelineStageArgs* outArgs); #endif // [ByteCode, ErrorText] // If successful, ByteCode != nullptr, otherwise, ErrorText contains the reason for failure. using ByteCodeResult = std::tuple, SkString>; ByteCodeResult toByteCode(const void* inputs); static void RegisterFlattenables(); ~SkRuntimeEffect(); private: SkRuntimeEffect(SkString sksl, std::unique_ptr baseProgram, std::vector&& inAndUniformVars, std::vector&& children, size_t uniformSize); using SpecializeResult = std::tuple, SkString>; SpecializeResult specialize(SkSL::Program& baseProgram, const void* inputs, const SkSL::SharedCompiler&); uint32_t fHash; SkString fSkSL; std::unique_ptr fBaseProgram; std::vector fInAndUniformVars; std::vector fChildren; size_t fUniformSize; }; #endif