blob: 8915821417fe3595d5499d8957fa528065e89762 [file] [log] [blame]
ethannicholasb3058bd2016-07-01 08:22:01 -07001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
Mike Klein6ad99092016-10-26 10:35:22 -04007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/sksl/SkSLCompiler.h"
ethannicholasb3058bd2016-07-01 08:22:01 -07009
John Stilesfbd050b2020-08-03 13:21:46 -040010#include <memory>
John Stilesb8e010c2020-08-11 18:05:39 -040011#include <unordered_set>
John Stilesfbd050b2020-08-03 13:21:46 -040012
John Stiles270cec22021-02-17 12:59:36 -050013#include "src/core/SkScopeExit.h"
Leon Scrogginsb66214e2021-02-11 17:14:18 -050014#include "src/core/SkTraceEvent.h"
John Stilesb92641c2020-08-31 18:09:01 -040015#include "src/sksl/SkSLAnalysis.h"
John Stilesf3a28db2021-03-10 23:00:47 -050016#include "src/sksl/SkSLConstantFolder.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "src/sksl/SkSLIRGenerator.h"
Brian Osman00185012021-02-04 16:07:11 -050018#include "src/sksl/SkSLOperators.h"
John Stiles270cec22021-02-17 12:59:36 -050019#include "src/sksl/SkSLProgramSettings.h"
Ethan Nicholasc18bb512020-07-28 14:46:53 -040020#include "src/sksl/SkSLRehydrator.h"
John Stiles3738ef52021-04-13 10:41:57 -040021#include "src/sksl/codegen/SkSLCPPCodeGenerator.h"
John Stiles82ab3402021-04-13 17:13:03 -040022#include "src/sksl/codegen/SkSLDSLCPPCodeGenerator.h"
John Stiles3738ef52021-04-13 10:41:57 -040023#include "src/sksl/codegen/SkSLGLSLCodeGenerator.h"
24#include "src/sksl/codegen/SkSLHCodeGenerator.h"
25#include "src/sksl/codegen/SkSLMetalCodeGenerator.h"
26#include "src/sksl/codegen/SkSLSPIRVCodeGenerator.h"
27#include "src/sksl/codegen/SkSLSPIRVtoHLSL.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050028#include "src/sksl/ir/SkSLEnum.h"
29#include "src/sksl/ir/SkSLExpression.h"
30#include "src/sksl/ir/SkSLExpressionStatement.h"
31#include "src/sksl/ir/SkSLFunctionCall.h"
32#include "src/sksl/ir/SkSLIntLiteral.h"
33#include "src/sksl/ir/SkSLModifiersDeclaration.h"
34#include "src/sksl/ir/SkSLNop.h"
35#include "src/sksl/ir/SkSLSymbolTable.h"
36#include "src/sksl/ir/SkSLTernaryExpression.h"
37#include "src/sksl/ir/SkSLUnresolvedFunction.h"
38#include "src/sksl/ir/SkSLVarDeclarations.h"
John Stilese6150002020-10-05 12:03:53 -040039#include "src/utils/SkBitSet.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070040
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -040041#include <fstream>
42
Ethan Nicholasa11035b2019-11-26 16:27:47 -050043#if !defined(SKSL_STANDALONE) & SK_SUPPORT_GPU
44#include "include/gpu/GrContextOptions.h"
45#include "src/gpu/GrShaderCaps.h"
46#endif
47
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -040048#ifdef SK_ENABLE_SPIRV_VALIDATION
49#include "spirv-tools/libspirv.hpp"
50#endif
51
Brian Osman3d87e9f2020-10-08 11:50:22 -040052#if defined(SKSL_STANDALONE)
Ethan Nicholasc18bb512020-07-28 14:46:53 -040053
Brian Osman3d87e9f2020-10-08 11:50:22 -040054// In standalone mode, we load the textual sksl source files. GN generates or copies these files
55// to the skslc executable directory. The "data" in this mode is just the filename.
56#define MODULE_DATA(name) MakeModulePath("sksl_" #name ".sksl")
57
58#else
59
60// At runtime, we load the dehydrated sksl data files. The data is a (pointer, size) pair.
Ethan Nicholasc18bb512020-07-28 14:46:53 -040061#include "src/sksl/generated/sksl_fp.dehydrated.sksl"
62#include "src/sksl/generated/sksl_frag.dehydrated.sksl"
63#include "src/sksl/generated/sksl_geom.dehydrated.sksl"
64#include "src/sksl/generated/sksl_gpu.dehydrated.sksl"
Brian Osmanb06301e2020-11-06 11:45:36 -050065#include "src/sksl/generated/sksl_public.dehydrated.sksl"
Brian Osmancbb60bd2021-04-12 09:49:20 -040066#include "src/sksl/generated/sksl_rt_colorfilter.dehydrated.sksl"
67#include "src/sksl/generated/sksl_rt_shader.dehydrated.sksl"
Ethan Nicholasc18bb512020-07-28 14:46:53 -040068#include "src/sksl/generated/sksl_vert.dehydrated.sksl"
69
Brian Osman3d87e9f2020-10-08 11:50:22 -040070#define MODULE_DATA(name) MakeModuleData(SKSL_INCLUDE_sksl_##name,\
71 SKSL_INCLUDE_sksl_##name##_LENGTH)
Ethan Nicholasc18bb512020-07-28 14:46:53 -040072
73#endif
Ethan Nicholas0d997662019-04-08 09:46:01 -040074
ethannicholasb3058bd2016-07-01 08:22:01 -070075namespace SkSL {
76
John Stiles7247b482021-03-08 10:40:35 -050077// These flags allow tools like Viewer or Nanobench to override the compiler's ProgramSettings.
John Stiles2ee4d7a2021-03-30 10:30:47 -040078Compiler::OverrideFlag Compiler::sOptimizer = OverrideFlag::kDefault;
79Compiler::OverrideFlag Compiler::sInliner = OverrideFlag::kDefault;
John Stiles8ef4d6c2021-03-05 16:01:45 -050080
John Stiles47c0a742021-02-09 09:30:35 -050081using RefKind = VariableReference::RefKind;
82
Brian Osman88cda172020-10-09 12:05:16 -040083class AutoSource {
84public:
85 AutoSource(Compiler* compiler, const String* source)
86 : fCompiler(compiler), fOldSource(fCompiler->fSource) {
87 fCompiler->fSource = source;
88 }
89
90 ~AutoSource() { fCompiler->fSource = fOldSource; }
91
92 Compiler* fCompiler;
93 const String* fOldSource;
94};
95
John Stilesa935c3f2021-02-25 10:35:49 -050096class AutoProgramConfig {
97public:
98 AutoProgramConfig(std::shared_ptr<Context>& context, ProgramConfig* config)
99 : fContext(context.get()) {
100 SkASSERT(!fContext->fConfig);
101 fContext->fConfig = config;
102 }
103
104 ~AutoProgramConfig() {
105 fContext->fConfig = nullptr;
106 }
107
108 Context* fContext;
109};
110
John Stilesd6a5f4492021-02-11 15:46:11 -0500111Compiler::Compiler(const ShaderCapsClass* caps)
John Stilesc1a98b82021-02-24 13:35:02 -0500112 : fContext(std::make_shared<Context>(/*errors=*/*this, *caps))
John Stiles7b920442020-12-17 10:43:41 -0500113 , fInliner(fContext.get())
Brian Osman0006ad02020-11-18 15:38:39 -0500114 , fErrorCount(0) {
John Stilesc1a98b82021-02-24 13:35:02 -0500115 SkASSERT(caps);
John Stiles7c3515b2020-10-16 18:38:39 -0400116 fRootSymbolTable = std::make_shared<SymbolTable>(this, /*builtin=*/true);
Brian Osmanb06301e2020-11-06 11:45:36 -0500117 fPrivateSymbolTable = std::make_shared<SymbolTable>(fRootSymbolTable, /*builtin=*/true);
John Stilesc1a98b82021-02-24 13:35:02 -0500118 fIRGenerator = std::make_unique<IRGenerator>(fContext.get());
ethannicholasb3058bd2016-07-01 08:22:01 -0700119
John Stiles54e7c052021-01-11 14:22:36 -0500120#define TYPE(t) fContext->fTypes.f ## t .get()
ethannicholasb3058bd2016-07-01 08:22:01 -0700121
Brian Osmanb06301e2020-11-06 11:45:36 -0500122 const SkSL::Symbol* rootTypes[] = {
123 TYPE(Void),
Brian Salomonbf7b6202016-11-11 16:08:03 -0500124
Brian Osmanb06301e2020-11-06 11:45:36 -0500125 TYPE( Float), TYPE( Float2), TYPE( Float3), TYPE( Float4),
126 TYPE( Half), TYPE( Half2), TYPE( Half3), TYPE( Half4),
127 TYPE( Int), TYPE( Int2), TYPE( Int3), TYPE( Int4),
Brian Osmanb06301e2020-11-06 11:45:36 -0500128 TYPE( Bool), TYPE( Bool2), TYPE( Bool3), TYPE( Bool4),
Brian Salomon2a51de82016-11-16 12:06:01 -0500129
Brian Osmanc0f2b642020-12-22 13:35:55 -0500130 TYPE(Float2x2), TYPE(Float3x3), TYPE(Float4x4),
Brian Osmanc63f4312020-12-23 11:44:14 -0500131 TYPE( Half2x2), TYPE( Half3x3), TYPE(Half4x4),
Greg Daniel64773e62016-11-22 09:44:03 -0500132
Brian Osmanc63f4312020-12-23 11:44:14 -0500133 TYPE(SquareMat), TYPE(SquareHMat),
ethannicholasb3058bd2016-07-01 08:22:01 -0700134
Brian Osman20fad322020-12-23 12:42:33 -0500135 TYPE(GenType), TYPE(GenHType), TYPE(GenIType), TYPE(GenBType),
136 TYPE(Vec), TYPE(HVec), TYPE(IVec), TYPE(BVec),
Brian Osmanb06301e2020-11-06 11:45:36 -0500137
Brian Osman14d00962021-04-02 17:04:35 -0400138 TYPE(ColorFilter),
139 TYPE(Shader),
Brian Osmanb06301e2020-11-06 11:45:36 -0500140 };
141
142 const SkSL::Symbol* privateTypes[] = {
Brian Osman20fad322020-12-23 12:42:33 -0500143 TYPE( UInt), TYPE( UInt2), TYPE( UInt3), TYPE( UInt4),
144 TYPE( Short), TYPE( Short2), TYPE( Short3), TYPE( Short4),
145 TYPE(UShort), TYPE(UShort2), TYPE(UShort3), TYPE(UShort4),
Brian Osman20fad322020-12-23 12:42:33 -0500146
147 TYPE(GenUType), TYPE(UVec),
Ethan Nicholas722c83e2021-05-04 11:39:30 -0400148 TYPE(SVec), TYPE(USVec),
Brian Osman20fad322020-12-23 12:42:33 -0500149
Brian Osmanc0f2b642020-12-22 13:35:55 -0500150 TYPE(Float2x3), TYPE(Float2x4),
151 TYPE(Float3x2), TYPE(Float3x4),
152 TYPE(Float4x2), TYPE(Float4x3),
153
Brian Osmanc63f4312020-12-23 11:44:14 -0500154 TYPE(Half2x3), TYPE(Half2x4),
155 TYPE(Half3x2), TYPE(Half3x4),
156 TYPE(Half4x2), TYPE(Half4x3),
157
Brian Osmanc0f2b642020-12-22 13:35:55 -0500158 TYPE(Mat), TYPE(HMat),
159
Brian Osmanb06301e2020-11-06 11:45:36 -0500160 TYPE(Sampler1D), TYPE(Sampler2D), TYPE(Sampler3D),
161 TYPE(SamplerExternalOES),
Brian Osmanb06301e2020-11-06 11:45:36 -0500162 TYPE(Sampler2DRect),
Brian Osmanb06301e2020-11-06 11:45:36 -0500163
164 TYPE(ISampler2D),
Brian Osmanb06301e2020-11-06 11:45:36 -0500165 TYPE(SubpassInput), TYPE(SubpassInputMS),
166
Brian Osmanb06301e2020-11-06 11:45:36 -0500167 TYPE(Sampler),
168 TYPE(Texture2D),
Brian Osman14d00962021-04-02 17:04:35 -0400169
170 TYPE(FragmentProcessor),
Brian Osmanb06301e2020-11-06 11:45:36 -0500171 };
172
173 for (const SkSL::Symbol* type : rootTypes) {
174 fRootSymbolTable->addWithoutOwnership(type);
175 }
176 for (const SkSL::Symbol* type : privateTypes) {
177 fPrivateSymbolTable->addWithoutOwnership(type);
178 }
179
180#undef TYPE
ethannicholasb3058bd2016-07-01 08:22:01 -0700181
Brian Osman3887a012020-09-30 13:22:27 -0400182 // sk_Caps is "builtin", but all references to it are resolved to Settings, so we don't need to
183 // treat it as builtin (ie, no need to clone it into the Program).
Brian Osmanb06301e2020-11-06 11:45:36 -0500184 fPrivateSymbolTable->add(
185 std::make_unique<Variable>(/*offset=*/-1,
John Stilesf2872e62021-05-04 11:38:43 -0400186 fIRGenerator->modifiersPool().add(Modifiers()),
Brian Osmanb06301e2020-11-06 11:45:36 -0500187 "sk_Caps",
John Stiles54e7c052021-01-11 14:22:36 -0500188 fContext->fTypes.fSkCaps.get(),
Brian Osmanb06301e2020-11-06 11:45:36 -0500189 /*builtin=*/false,
190 Variable::Storage::kGlobal));
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500191
Brian Osman3d87e9f2020-10-08 11:50:22 -0400192 fRootModule = {fRootSymbolTable, /*fIntrinsics=*/nullptr};
Brian Osmanb06301e2020-11-06 11:45:36 -0500193 fPrivateModule = {fPrivateSymbolTable, /*fIntrinsics=*/nullptr};
ethannicholasb3058bd2016-07-01 08:22:01 -0700194}
195
John Stilesdd13dba2020-10-29 10:45:34 -0400196Compiler::~Compiler() {}
ethannicholasb3058bd2016-07-01 08:22:01 -0700197
Brian Osman56269982020-11-20 12:38:07 -0500198const ParsedModule& Compiler::loadGPUModule() {
199 if (!fGPUModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500200 fGPUModule = this->parseModule(ProgramKind::kFragment, MODULE_DATA(gpu), fPrivateModule);
Brian Osman56269982020-11-20 12:38:07 -0500201 }
202 return fGPUModule;
203}
204
205const ParsedModule& Compiler::loadFragmentModule() {
206 if (!fFragmentModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500207 fFragmentModule = this->parseModule(ProgramKind::kFragment, MODULE_DATA(frag),
Brian Osman56269982020-11-20 12:38:07 -0500208 this->loadGPUModule());
209 }
210 return fFragmentModule;
211}
212
213const ParsedModule& Compiler::loadVertexModule() {
214 if (!fVertexModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500215 fVertexModule = this->parseModule(ProgramKind::kVertex, MODULE_DATA(vert),
Brian Osman56269982020-11-20 12:38:07 -0500216 this->loadGPUModule());
217 }
218 return fVertexModule;
219}
220
Brian Osman88cda172020-10-09 12:05:16 -0400221const ParsedModule& Compiler::loadGeometryModule() {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400222 if (!fGeometryModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500223 fGeometryModule = this->parseModule(ProgramKind::kGeometry, MODULE_DATA(geom),
Brian Osman56269982020-11-20 12:38:07 -0500224 this->loadGPUModule());
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400225 }
Brian Osman88cda172020-10-09 12:05:16 -0400226 return fGeometryModule;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400227}
228
Brian Osman88cda172020-10-09 12:05:16 -0400229const ParsedModule& Compiler::loadFPModule() {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400230 if (!fFPModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500231 fFPModule = this->parseModule(ProgramKind::kFragmentProcessor, MODULE_DATA(fp),
Brian Osman56269982020-11-20 12:38:07 -0500232 this->loadGPUModule());
Brian Osman8e2ef022020-09-30 13:26:43 -0400233 }
Brian Osman88cda172020-10-09 12:05:16 -0400234 return fFPModule;
Brian Osman8e2ef022020-09-30 13:26:43 -0400235}
236
Brian Osmanb06301e2020-11-06 11:45:36 -0500237const ParsedModule& Compiler::loadPublicModule() {
238 if (!fPublicModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500239 fPublicModule = this->parseModule(ProgramKind::kGeneric, MODULE_DATA(public), fRootModule);
Brian Osmanb06301e2020-11-06 11:45:36 -0500240 }
241 return fPublicModule;
242}
243
Brian Osmancbb60bd2021-04-12 09:49:20 -0400244static void add_glsl_type_aliases(SkSL::SymbolTable* symbols, const SkSL::BuiltinTypes& types) {
245 // Add some aliases to the runtime effect modules so that it's friendlier, and more like GLSL
246 symbols->addAlias("vec2", types.fFloat2.get());
247 symbols->addAlias("vec3", types.fFloat3.get());
248 symbols->addAlias("vec4", types.fFloat4.get());
249
250 symbols->addAlias("ivec2", types.fInt2.get());
251 symbols->addAlias("ivec3", types.fInt3.get());
252 symbols->addAlias("ivec4", types.fInt4.get());
253
254 symbols->addAlias("bvec2", types.fBool2.get());
255 symbols->addAlias("bvec3", types.fBool3.get());
256 symbols->addAlias("bvec4", types.fBool4.get());
257
258 symbols->addAlias("mat2", types.fFloat2x2.get());
259 symbols->addAlias("mat3", types.fFloat3x3.get());
260 symbols->addAlias("mat4", types.fFloat4x4.get());
261}
262
Brian Osmancbb60bd2021-04-12 09:49:20 -0400263const ParsedModule& Compiler::loadRuntimeColorFilterModule() {
264 if (!fRuntimeColorFilterModule.fSymbols) {
265 fRuntimeColorFilterModule = this->parseModule(ProgramKind::kRuntimeColorFilter,
266 MODULE_DATA(rt_colorfilter),
267 this->loadPublicModule());
268 add_glsl_type_aliases(fRuntimeColorFilterModule.fSymbols.get(), fContext->fTypes);
269 }
270 return fRuntimeColorFilterModule;
271}
272
273const ParsedModule& Compiler::loadRuntimeShaderModule() {
274 if (!fRuntimeShaderModule.fSymbols) {
275 fRuntimeShaderModule = this->parseModule(
276 ProgramKind::kRuntimeShader, MODULE_DATA(rt_shader), this->loadPublicModule());
277 add_glsl_type_aliases(fRuntimeShaderModule.fSymbols.get(), fContext->fTypes);
278 }
279 return fRuntimeShaderModule;
280}
281
John Stilesdbd4e6f2021-02-16 13:29:15 -0500282const ParsedModule& Compiler::moduleForProgramKind(ProgramKind kind) {
Brian Osman88cda172020-10-09 12:05:16 -0400283 switch (kind) {
Brian Osmancbb60bd2021-04-12 09:49:20 -0400284 case ProgramKind::kVertex: return this->loadVertexModule(); break;
285 case ProgramKind::kFragment: return this->loadFragmentModule(); break;
286 case ProgramKind::kGeometry: return this->loadGeometryModule(); break;
287 case ProgramKind::kFragmentProcessor: return this->loadFPModule(); break;
Brian Osmancbb60bd2021-04-12 09:49:20 -0400288 case ProgramKind::kRuntimeColorFilter: return this->loadRuntimeColorFilterModule(); break;
289 case ProgramKind::kRuntimeShader: return this->loadRuntimeShaderModule(); break;
290 case ProgramKind::kGeneric: return this->loadPublicModule(); break;
Brian Osman88cda172020-10-09 12:05:16 -0400291 }
292 SkUNREACHABLE;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400293}
294
John Stilesdbd4e6f2021-02-16 13:29:15 -0500295LoadedModule Compiler::loadModule(ProgramKind kind,
Brian Osman3d87e9f2020-10-08 11:50:22 -0400296 ModuleData data,
John Stilesa935c3f2021-02-25 10:35:49 -0500297 std::shared_ptr<SymbolTable> base,
298 bool dehydrate) {
299 if (dehydrate) {
300 // NOTE: This is a workaround. When dehydrating includes, skslc doesn't know which module
301 // it's preparing, nor what the correct base module is. We can't use 'Root', because many
302 // GPU intrinsics reference private types, like samplers or textures. Today, 'Private' does
303 // contain the union of all known types, so this is safe. If we ever have types that only
304 // exist in 'Public' (for example), this logic needs to be smarter (by choosing the correct
305 // base for the module we're compiling).
Brian Osmanb06301e2020-11-06 11:45:36 -0500306 base = fPrivateSymbolTable;
Brian Osman3d87e9f2020-10-08 11:50:22 -0400307 }
John Stilesa935c3f2021-02-25 10:35:49 -0500308 SkASSERT(base);
309
310 // Built-in modules always use default program settings.
311 ProgramConfig config;
312 config.fKind = kind;
313 config.fSettings.fReplaceSettings = !dehydrate;
314 AutoProgramConfig autoConfig(fContext, &config);
Brian Osman3d87e9f2020-10-08 11:50:22 -0400315
316#if defined(SKSL_STANDALONE)
317 SkASSERT(data.fPath);
318 std::ifstream in(data.fPath);
John Stilesd51c9792021-03-18 11:40:14 -0400319 String text{std::istreambuf_iterator<char>(in), std::istreambuf_iterator<char>()};
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400320 if (in.rdstate()) {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400321 printf("error reading %s\n", data.fPath);
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400322 abort();
323 }
Brian Osmane498b3c2020-09-23 14:42:11 -0400324 const String* source = fRootSymbolTable->takeOwnershipOfString(std::move(text));
Brian Osman88cda172020-10-09 12:05:16 -0400325 AutoSource as(this, source);
John Stilesd1204642021-02-17 16:30:02 -0500326
Brian Osman88cda172020-10-09 12:05:16 -0400327 ParsedModule baseModule = {base, /*fIntrinsics=*/nullptr};
John Stilesd1204642021-02-17 16:30:02 -0500328 IRGenerator::IRBundle ir = fIRGenerator->convertProgram(baseModule, /*isBuiltinCode=*/true,
329 source->c_str(), source->length(),
330 /*externalFunctions=*/nullptr);
Brian Osman133724c2020-10-28 14:14:39 -0400331 SkASSERT(ir.fSharedElements.empty());
Brian Osman0006ad02020-11-18 15:38:39 -0500332 LoadedModule module = { kind, std::move(ir.fSymbolTable), std::move(ir.fElements) };
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400333 if (this->fErrorCount) {
334 printf("Unexpected errors: %s\n", this->fErrorText.c_str());
Brian Osman3d87e9f2020-10-08 11:50:22 -0400335 SkDEBUGFAILF("%s %s\n", data.fPath, this->fErrorText.c_str());
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400336 }
Brian Osman88cda172020-10-09 12:05:16 -0400337 fModifiers.push_back(std::move(ir.fModifiers));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400338#else
339 SkASSERT(data.fData && (data.fSize != 0));
John Stilesf2872e62021-05-04 11:38:43 -0400340 Rehydrator rehydrator(fContext.get(), fIRGenerator->fModifiers.get(), base,
Brian Osman3d87e9f2020-10-08 11:50:22 -0400341 data.fData, data.fSize);
Brian Osman0006ad02020-11-18 15:38:39 -0500342 LoadedModule module = { kind, rehydrator.symbolTable(), rehydrator.elements() };
Brian Osman3d87e9f2020-10-08 11:50:22 -0400343 fModifiers.push_back(fIRGenerator->releaseModifiers());
344#endif
345
346 return module;
347}
348
John Stilesdbd4e6f2021-02-16 13:29:15 -0500349ParsedModule Compiler::parseModule(ProgramKind kind, ModuleData data, const ParsedModule& base) {
John Stilesa935c3f2021-02-25 10:35:49 -0500350 LoadedModule module = this->loadModule(kind, data, base.fSymbols, /*dehydrate=*/false);
Brian Osman0006ad02020-11-18 15:38:39 -0500351 this->optimize(module);
Brian Osman3d87e9f2020-10-08 11:50:22 -0400352
353 // For modules that just declare (but don't define) intrinsic functions, there will be no new
354 // program elements. In that case, we can share our parent's intrinsic map:
Brian Osman0006ad02020-11-18 15:38:39 -0500355 if (module.fElements.empty()) {
356 return {module.fSymbols, base.fIntrinsics};
Brian Osman3d87e9f2020-10-08 11:50:22 -0400357 }
358
359 auto intrinsics = std::make_shared<IRIntrinsicMap>(base.fIntrinsics.get());
360
361 // Now, transfer all of the program elements to an intrinsic map. This maps certain types of
362 // global objects to the declaring ProgramElement.
Brian Osman0006ad02020-11-18 15:38:39 -0500363 for (std::unique_ptr<ProgramElement>& element : module.fElements) {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400364 switch (element->kind()) {
365 case ProgramElement::Kind::kFunction: {
366 const FunctionDefinition& f = element->as<FunctionDefinition>();
Ethan Nicholas0a5d0962020-10-14 13:33:18 -0400367 SkASSERT(f.declaration().isBuiltin());
368 intrinsics->insertOrDie(f.declaration().description(), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400369 break;
370 }
John Stiles569249b2020-11-03 12:18:22 -0500371 case ProgramElement::Kind::kFunctionPrototype: {
372 // These are already in the symbol table.
373 break;
374 }
Brian Osman3d87e9f2020-10-08 11:50:22 -0400375 case ProgramElement::Kind::kEnum: {
376 const Enum& e = element->as<Enum>();
377 SkASSERT(e.isBuiltin());
378 intrinsics->insertOrDie(e.typeName(), std::move(element));
379 break;
380 }
381 case ProgramElement::Kind::kGlobalVar: {
Ethan Nicholasc51f33e2020-10-13 13:49:44 -0400382 const GlobalVarDeclaration& global = element->as<GlobalVarDeclaration>();
383 const Variable& var = global.declaration()->as<VarDeclaration>().var();
384 SkASSERT(var.isBuiltin());
385 intrinsics->insertOrDie(var.name(), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400386 break;
387 }
388 case ProgramElement::Kind::kInterfaceBlock: {
Ethan Nicholaseaf47882020-10-15 10:10:08 -0400389 const Variable& var = element->as<InterfaceBlock>().variable();
390 SkASSERT(var.isBuiltin());
391 intrinsics->insertOrDie(var.name(), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400392 break;
393 }
394 default:
395 printf("Unsupported element: %s\n", element->description().c_str());
396 SkASSERT(false);
397 break;
398 }
399 }
400
Brian Osman0006ad02020-11-18 15:38:39 -0500401 return {module.fSymbols, std::move(intrinsics)};
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400402}
403
Brian Osman32d53552020-09-23 13:55:20 -0400404std::unique_ptr<Program> Compiler::convertProgram(
John Stilesdbd4e6f2021-02-16 13:29:15 -0500405 ProgramKind kind,
Brian Osman32d53552020-09-23 13:55:20 -0400406 String text,
407 const Program::Settings& settings,
Brian Osmanbe0b3b72021-01-06 14:27:35 -0500408 const std::vector<std::unique_ptr<ExternalFunction>>* externalFunctions) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400409 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::convertProgram");
Leon Scrogginsb66214e2021-02-11 17:14:18 -0500410
John Stilesdbd4e6f2021-02-16 13:29:15 -0500411 SkASSERT(!externalFunctions || (kind == ProgramKind::kGeneric));
Ethan Nicholas91164d12019-05-15 15:29:54 -0400412
Brian Osman0006ad02020-11-18 15:38:39 -0500413 // Loading and optimizing our base module might reset the inliner, so do that first,
414 // *then* configure the inliner with the settings for this program.
415 const ParsedModule& baseModule = this->moduleForProgramKind(kind);
416
John Stiles270cec22021-02-17 12:59:36 -0500417 // Update our context to point to the program configuration for the duration of compilation.
418 auto config = std::make_unique<ProgramConfig>(ProgramConfig{kind, settings});
John Stilesa935c3f2021-02-25 10:35:49 -0500419 AutoProgramConfig autoConfig(fContext, config.get());
John Stiles270cec22021-02-17 12:59:36 -0500420
John Stiles2ee4d7a2021-03-30 10:30:47 -0400421 // Honor our optimization-override flags.
422 switch (sOptimizer) {
423 case OverrideFlag::kDefault:
424 break;
425 case OverrideFlag::kOff:
426 config->fSettings.fOptimize = false;
427 break;
428 case OverrideFlag::kOn:
429 config->fSettings.fOptimize = true;
430 break;
431 }
432
433 switch (sInliner) {
434 case OverrideFlag::kDefault:
435 break;
436 case OverrideFlag::kOff:
437 config->fSettings.fInlineThreshold = 0;
438 break;
439 case OverrideFlag::kOn:
440 if (config->fSettings.fInlineThreshold == 0) {
441 config->fSettings.fInlineThreshold = kDefaultInlineThreshold;
442 }
443 break;
444 }
John Stiles7247b482021-03-08 10:40:35 -0500445
446 // Disable optimization settings that depend on a parent setting which has been disabled.
John Stiles7247b482021-03-08 10:40:35 -0500447 config->fSettings.fInlineThreshold *= (int)config->fSettings.fOptimize;
John Stiles0bfeae62021-03-11 09:09:42 -0500448 config->fSettings.fRemoveDeadFunctions &= config->fSettings.fOptimize;
449 config->fSettings.fRemoveDeadVariables &= config->fSettings.fOptimize;
John Stiles7247b482021-03-08 10:40:35 -0500450
ethannicholasb3058bd2016-07-01 08:22:01 -0700451 fErrorText = "";
452 fErrorCount = 0;
John Stilesd1204642021-02-17 16:30:02 -0500453 fInliner.reset(fIRGenerator->fModifiers.get());
Brian Osman88cda172020-10-09 12:05:16 -0400454
455 // Not using AutoSource, because caller is likely to call errorText() if we fail to compile
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700456 std::unique_ptr<String> textPtr(new String(std::move(text)));
457 fSource = textPtr.get();
Brian Osman88cda172020-10-09 12:05:16 -0400458
John Stiles5c7bb322020-10-22 11:09:15 -0400459 // Enable node pooling while converting and optimizing the program for a performance boost.
460 // The Program will take ownership of the pool.
Brian Osman28f702c2021-02-02 11:52:07 -0500461 std::unique_ptr<Pool> pool;
John Stilesc1a98b82021-02-24 13:35:02 -0500462 if (fContext->fCaps.useNodePools()) {
Brian Osman28f702c2021-02-02 11:52:07 -0500463 pool = Pool::Create();
464 pool->attachToThread();
465 }
John Stilesd1204642021-02-17 16:30:02 -0500466 IRGenerator::IRBundle ir = fIRGenerator->convertProgram(baseModule, /*isBuiltinCode=*/false,
467 textPtr->c_str(), textPtr->size(),
468 externalFunctions);
John Stiles270cec22021-02-17 12:59:36 -0500469 auto program = std::make_unique<Program>(std::move(textPtr),
470 std::move(config),
John Stiles5c7bb322020-10-22 11:09:15 -0400471 fContext,
472 std::move(ir.fElements),
Brian Osman133724c2020-10-28 14:14:39 -0400473 std::move(ir.fSharedElements),
John Stiles5c7bb322020-10-22 11:09:15 -0400474 std::move(ir.fModifiers),
475 std::move(ir.fSymbolTable),
476 std::move(pool),
477 ir.fInputs);
478 bool success = false;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500479 if (fErrorCount) {
John Stiles5c7bb322020-10-22 11:09:15 -0400480 // Do not return programs that failed to compile.
John Stiles7247b482021-03-08 10:40:35 -0500481 } else if (!this->optimize(*program)) {
John Stiles5c7bb322020-10-22 11:09:15 -0400482 // Do not return programs that failed to optimize.
483 } else {
484 // We have a successful program!
485 success = true;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500486 }
John Stiles5c7bb322020-10-22 11:09:15 -0400487
Brian Osman28f702c2021-02-02 11:52:07 -0500488 if (program->fPool) {
489 program->fPool->detachFromThread();
490 }
John Stiles5c7bb322020-10-22 11:09:15 -0400491 return success ? std::move(program) : nullptr;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500492}
493
John Stilesbb1505f2021-02-12 09:17:53 -0500494void Compiler::verifyStaticTests(const Program& program) {
495 class StaticTestVerifier : public ProgramVisitor {
496 public:
497 StaticTestVerifier(ErrorReporter* r) : fReporter(r) {}
498
499 using ProgramVisitor::visitProgramElement;
500
501 bool visitStatement(const Statement& stmt) override {
502 switch (stmt.kind()) {
503 case Statement::Kind::kIf:
504 if (stmt.as<IfStatement>().isStatic()) {
505 fReporter->error(stmt.fOffset, "static if has non-static test");
506 }
507 break;
508
509 case Statement::Kind::kSwitch:
510 if (stmt.as<SwitchStatement>().isStatic()) {
511 fReporter->error(stmt.fOffset, "static switch has non-static test");
512 }
513 break;
514
515 default:
516 break;
517 }
518 return INHERITED::visitStatement(stmt);
519 }
520
John Stiles59e34562021-02-12 16:56:39 -0500521 bool visitExpression(const Expression&) override {
522 // We aren't looking for anything inside an Expression, so skip them entirely.
523 return false;
524 }
525
John Stilesbb1505f2021-02-12 09:17:53 -0500526 private:
527 using INHERITED = ProgramVisitor;
528 ErrorReporter* fReporter;
529 };
530
531 // If invalid static tests are permitted, we don't need to check anything.
John Stilesd1204642021-02-17 16:30:02 -0500532 if (fContext->fConfig->fSettings.fPermitInvalidStaticTests) {
John Stilesbb1505f2021-02-12 09:17:53 -0500533 return;
534 }
535
536 // Check all of the program's owned elements. (Built-in elements are assumed to be valid.)
537 StaticTestVerifier visitor{this};
538 for (const std::unique_ptr<ProgramElement>& element : program.ownedElements()) {
539 if (element->is<FunctionDefinition>()) {
540 visitor.visitProgramElement(*element);
541 }
542 }
543}
544
Brian Osman0006ad02020-11-18 15:38:39 -0500545bool Compiler::optimize(LoadedModule& module) {
546 SkASSERT(!fErrorCount);
Brian Osman0006ad02020-11-18 15:38:39 -0500547
John Stiles270cec22021-02-17 12:59:36 -0500548 // Create a temporary program configuration with default settings.
549 ProgramConfig config;
550 config.fKind = module.fKind;
John Stilesa935c3f2021-02-25 10:35:49 -0500551 AutoProgramConfig autoConfig(fContext, &config);
John Stiles270cec22021-02-17 12:59:36 -0500552
John Stilesd1204642021-02-17 16:30:02 -0500553 // Reset the Inliner.
554 fInliner.reset(fModifiers.back().get());
John Stiles270cec22021-02-17 12:59:36 -0500555
556 std::unique_ptr<ProgramUsage> usage = Analysis::GetUsage(module);
Brian Osman0006ad02020-11-18 15:38:39 -0500557
558 while (fErrorCount == 0) {
Brian Osman0006ad02020-11-18 15:38:39 -0500559 // Perform inline-candidate analysis and inline any functions deemed suitable.
John Stilesf3a28db2021-03-10 23:00:47 -0500560 if (!fInliner.analyze(module.fElements, module.fSymbols, usage.get())) {
Brian Osman0006ad02020-11-18 15:38:39 -0500561 break;
562 }
563 }
564 return fErrorCount == 0;
565}
566
John Stiles0bfeae62021-03-11 09:09:42 -0500567bool Compiler::removeDeadFunctions(Program& program, ProgramUsage* usage) {
568 bool madeChanges = false;
569
570 if (program.fConfig->fSettings.fRemoveDeadFunctions) {
571 auto isDeadFunction = [&](const ProgramElement* element) {
572 if (!element->is<FunctionDefinition>()) {
573 return false;
574 }
575 const FunctionDefinition& fn = element->as<FunctionDefinition>();
John Stilese8da4d22021-03-24 09:19:45 -0400576 if (fn.declaration().isMain() || usage->get(fn.declaration()) > 0) {
John Stiles0bfeae62021-03-11 09:09:42 -0500577 return false;
578 }
579 usage->remove(*element);
580 madeChanges = true;
581 return true;
582 };
583
584 program.fElements.erase(std::remove_if(program.fElements.begin(),
585 program.fElements.end(),
586 [&](const std::unique_ptr<ProgramElement>& element) {
587 return isDeadFunction(element.get());
588 }),
589 program.fElements.end());
590 program.fSharedElements.erase(std::remove_if(program.fSharedElements.begin(),
591 program.fSharedElements.end(),
592 isDeadFunction),
593 program.fSharedElements.end());
594 }
595 return madeChanges;
596}
597
598bool Compiler::removeDeadGlobalVariables(Program& program, ProgramUsage* usage) {
599 bool madeChanges = false;
600
601 if (program.fConfig->fSettings.fRemoveDeadVariables) {
602 auto isDeadVariable = [&](const ProgramElement* element) {
603 if (!element->is<GlobalVarDeclaration>()) {
604 return false;
605 }
606 const GlobalVarDeclaration& global = element->as<GlobalVarDeclaration>();
607 const VarDeclaration& varDecl = global.declaration()->as<VarDeclaration>();
608 if (!usage->isDead(varDecl.var())) {
609 return false;
610 }
611 madeChanges = true;
612 return true;
613 };
614
615 program.fElements.erase(std::remove_if(program.fElements.begin(),
616 program.fElements.end(),
617 [&](const std::unique_ptr<ProgramElement>& element) {
618 return isDeadVariable(element.get());
619 }),
620 program.fElements.end());
621 program.fSharedElements.erase(std::remove_if(program.fSharedElements.begin(),
622 program.fSharedElements.end(),
623 isDeadVariable),
624 program.fSharedElements.end());
625 }
626 return madeChanges;
627}
628
John Stiles26541872021-03-16 12:19:54 -0400629bool Compiler::removeDeadLocalVariables(Program& program, ProgramUsage* usage) {
630 class DeadLocalVariableEliminator : public ProgramWriter {
631 public:
632 DeadLocalVariableEliminator(const Context& context, ProgramUsage* usage)
633 : fContext(context)
634 , fUsage(usage) {}
635
636 using ProgramWriter::visitProgramElement;
637
638 bool visitExpressionPtr(std::unique_ptr<Expression>& expr) override {
639 // We don't need to look inside expressions at all.
640 return false;
641 }
642
643 bool visitStatementPtr(std::unique_ptr<Statement>& stmt) override {
644 if (stmt->is<VarDeclaration>()) {
645 VarDeclaration& varDecl = stmt->as<VarDeclaration>();
646 const Variable* var = &varDecl.var();
647 ProgramUsage::VariableCounts* counts = fUsage->fVariableCounts.find(var);
648 SkASSERT(counts);
649 SkASSERT(counts->fDeclared);
650 if (CanEliminate(var, *counts)) {
651 if (var->initialValue()) {
652 // The variable has an initial-value expression, which might have side
653 // effects. ExpressionStatement::Make will preserve side effects, but
654 // replaces pure expressions with Nop.
655 fUsage->remove(stmt.get());
656 stmt = ExpressionStatement::Make(fContext, std::move(varDecl.value()));
657 fUsage->add(stmt.get());
658 } else {
659 // The variable has no initial-value and can be cleanly eliminated.
660 fUsage->remove(stmt.get());
661 stmt = std::make_unique<Nop>();
662 }
663 fMadeChanges = true;
664 }
665 return false;
666 }
667 return INHERITED::visitStatementPtr(stmt);
668 }
669
670 static bool CanEliminate(const Variable* var, const ProgramUsage::VariableCounts& counts) {
671 if (!counts.fDeclared || counts.fRead || var->storage() != VariableStorage::kLocal) {
672 return false;
673 }
674 if (var->initialValue()) {
675 SkASSERT(counts.fWrite >= 1);
676 return counts.fWrite == 1;
677 } else {
678 return counts.fWrite == 0;
679 }
680 }
681
682 bool fMadeChanges = false;
683 const Context& fContext;
684 ProgramUsage* fUsage;
685
686 using INHERITED = ProgramWriter;
687 };
688
689 DeadLocalVariableEliminator visitor{*fContext, usage};
690
691 if (program.fConfig->fSettings.fRemoveDeadVariables) {
692 for (auto& [var, counts] : usage->fVariableCounts) {
693 if (DeadLocalVariableEliminator::CanEliminate(var, counts)) {
694 // This program contains at least one dead local variable.
695 // Scan the program for any dead local variables and eliminate them all.
696 for (std::unique_ptr<ProgramElement>& pe : program.ownedElements()) {
697 if (pe->is<FunctionDefinition>()) {
698 visitor.visitProgramElement(*pe);
699 }
700 }
701 break;
702 }
703 }
704 }
705
706 return visitor.fMadeChanges;
707}
708
Ethan Nicholas00543112018-07-31 09:44:36 -0400709bool Compiler::optimize(Program& program) {
John Stiles7247b482021-03-08 10:40:35 -0500710 // The optimizer only needs to run when it is enabled.
711 if (!program.fConfig->fSettings.fOptimize) {
712 return true;
713 }
714
Ethan Nicholas00543112018-07-31 09:44:36 -0400715 SkASSERT(!fErrorCount);
Brian Osman010ce6a2020-10-19 16:34:10 -0400716 ProgramUsage* usage = program.fUsage.get();
John Stiles7954d6c2020-09-01 10:53:02 -0400717
John Stilesb6664582021-03-19 09:46:00 -0400718 if (fErrorCount == 0) {
John Stiles87fc6572021-04-01 14:56:34 +0000719 // Run the inliner only once; it is expensive! Multiple passes can occasionally shake out
720 // more wins, but it's diminishing returns.
721 fInliner.analyze(program.ownedElements(), program.fSymbols, usage);
Ethan Nicholas34b19c52020-09-14 11:33:47 -0400722
John Stilesb6664582021-03-19 09:46:00 -0400723 while (this->removeDeadFunctions(program, usage)) {
724 // Removing dead functions may cause more functions to become unreferenced. Try again.
Ethan Nicholas34b19c52020-09-14 11:33:47 -0400725 }
John Stilesb6664582021-03-19 09:46:00 -0400726 while (this->removeDeadLocalVariables(program, usage)) {
727 // Removing dead variables may cause more variables to become unreferenced. Try again.
728 }
729 if (program.fConfig->fKind != ProgramKind::kFragmentProcessor) {
730 this->removeDeadGlobalVariables(program, usage);
Ethan Nicholas0dc80872019-02-08 15:46:24 -0500731 }
Ethan Nicholas00543112018-07-31 09:44:36 -0400732 }
John Stilesbb1505f2021-02-12 09:17:53 -0500733
734 if (fErrorCount == 0) {
735 this->verifyStaticTests(program);
736 }
737
Ethan Nicholas00543112018-07-31 09:44:36 -0400738 return fErrorCount == 0;
739}
740
Brian Osmanfb32ddf2019-06-18 10:14:20 -0400741#if defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
742
Ethan Nicholas00543112018-07-31 09:44:36 -0400743bool Compiler::toSPIRV(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400744 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toSPIRV");
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400745#ifdef SK_ENABLE_SPIRV_VALIDATION
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400746 StringStream buffer;
Brian Osman88cda172020-10-09 12:05:16 -0400747 AutoSource as(this, program.fSource.get());
Brian Osman8b43dad2020-10-09 13:31:42 -0400748 SPIRVCodeGenerator cg(fContext.get(), &program, this, &buffer);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400749 bool result = cg.generateCode();
John Stiles270cec22021-02-17 12:59:36 -0500750 if (result && program.fConfig->fSettings.fValidateSPIRV) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400751 spvtools::SpirvTools tools(SPV_ENV_VULKAN_1_0);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400752 const String& data = buffer.str();
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400753 SkASSERT(0 == data.size() % 4);
Brian Osman8d09d4a2020-11-24 15:51:06 -0500754 String errors;
755 auto dumpmsg = [&errors](spv_message_level_t, const char*, const spv_position_t&,
756 const char* m) {
757 errors.appendf("SPIR-V validation error: %s\n", m);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400758 };
759 tools.SetMessageConsumer(dumpmsg);
Brian Osman8d09d4a2020-11-24 15:51:06 -0500760
761 // Verify that the SPIR-V we produced is valid. At runtime, we will abort() with a message
762 // explaining the error. In standalone mode (skslc), we will send the message, plus the
763 // entire disassembled SPIR-V (for easier context & debugging) as *our* error message.
764 result = tools.Validate((const uint32_t*) data.c_str(), data.size() / 4);
765
766 if (!result) {
767#if defined(SKSL_STANDALONE)
768 // Convert the string-stream to a SPIR-V disassembly.
769 std::string disassembly;
770 if (tools.Disassemble((const uint32_t*)data.data(), data.size() / 4, &disassembly)) {
771 errors.append(disassembly);
772 }
773 this->error(-1, errors);
774#else
775 SkDEBUGFAILF("%s", errors.c_str());
776#endif
777 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400778 out.write(data.c_str(), data.size());
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400779 }
780#else
Brian Osman88cda172020-10-09 12:05:16 -0400781 AutoSource as(this, program.fSource.get());
Brian Osman8b43dad2020-10-09 13:31:42 -0400782 SPIRVCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500783 bool result = cg.generateCode();
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400784#endif
Ethan Nicholasce33f102016-12-09 17:22:59 -0500785 return result;
786}
787
Ethan Nicholas00543112018-07-31 09:44:36 -0400788bool Compiler::toSPIRV(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400789 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500790 bool result = this->toSPIRV(program, buffer);
791 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400792 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500793 }
794 return result;
795}
796
Ethan Nicholas00543112018-07-31 09:44:36 -0400797bool Compiler::toGLSL(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400798 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toGLSL");
Brian Osman88cda172020-10-09 12:05:16 -0400799 AutoSource as(this, program.fSource.get());
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400800 GLSLCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500801 bool result = cg.generateCode();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500802 return result;
803}
804
Ethan Nicholas00543112018-07-31 09:44:36 -0400805bool Compiler::toGLSL(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400806 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500807 bool result = this->toGLSL(program, buffer);
808 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400809 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500810 }
811 return result;
812}
813
Brian Osmanc0243912020-02-19 15:35:26 -0500814bool Compiler::toHLSL(Program& program, String* out) {
815 String spirv;
816 if (!this->toSPIRV(program, &spirv)) {
817 return false;
818 }
819
820 return SPIRVtoHLSL(spirv, out);
821}
822
Ethan Nicholas00543112018-07-31 09:44:36 -0400823bool Compiler::toMetal(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400824 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toMetal");
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400825 MetalCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholascc305772017-10-13 16:17:45 -0400826 bool result = cg.generateCode();
Ethan Nicholascc305772017-10-13 16:17:45 -0400827 return result;
828}
829
Ethan Nicholas00543112018-07-31 09:44:36 -0400830bool Compiler::toMetal(Program& program, String* out) {
Timothy Liangb8eeb802018-07-23 16:46:16 -0400831 StringStream buffer;
832 bool result = this->toMetal(program, buffer);
833 if (result) {
834 *out = buffer.str();
835 }
836 return result;
837}
838
Greg Daniela28ea672020-09-25 11:12:56 -0400839#if defined(SKSL_STANDALONE) || GR_TEST_UTILS
Ethan Nicholas00543112018-07-31 09:44:36 -0400840bool Compiler::toCPP(Program& program, String name, OutputStream& out) {
Brian Osman88cda172020-10-09 12:05:16 -0400841 AutoSource as(this, program.fSource.get());
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400842 CPPCodeGenerator cg(fContext.get(), &program, this, name, &out);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400843 bool result = cg.generateCode();
Ethan Nicholas762466e2017-06-29 10:03:38 -0400844 return result;
845}
846
John Stiles82ab3402021-04-13 17:13:03 -0400847bool Compiler::toDSLCPP(Program& program, String name, OutputStream& out) {
848 AutoSource as(this, program.fSource.get());
849 DSLCPPCodeGenerator cg(fContext.get(), &program, this, name, &out);
850 bool result = cg.generateCode();
851 return result;
852}
853
Ethan Nicholas00543112018-07-31 09:44:36 -0400854bool Compiler::toH(Program& program, String name, OutputStream& out) {
Brian Osman88cda172020-10-09 12:05:16 -0400855 AutoSource as(this, program.fSource.get());
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400856 HCodeGenerator cg(fContext.get(), &program, this, name, &out);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400857 bool result = cg.generateCode();
Ethan Nicholas00543112018-07-31 09:44:36 -0400858 return result;
859}
Greg Daniela28ea672020-09-25 11:12:56 -0400860#endif // defined(SKSL_STANDALONE) || GR_TEST_UTILS
Ethan Nicholas00543112018-07-31 09:44:36 -0400861
Ethan Nicholas2a479a52020-08-18 16:29:45 -0400862#endif // defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
Brian Osman2e29ab52019-09-20 12:19:11 -0400863
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700864Position Compiler::position(int offset) {
Ethan Nicholas3c729892020-12-07 12:47:17 -0500865 if (fSource && offset >= 0) {
866 int line = 1;
867 int column = 1;
868 for (int i = 0; i < offset; i++) {
869 if ((*fSource)[i] == '\n') {
870 ++line;
871 column = 1;
872 }
873 else {
874 ++column;
875 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700876 }
Ethan Nicholas3c729892020-12-07 12:47:17 -0500877 return Position(line, column);
878 } else {
879 return Position(-1, -1);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700880 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700881}
882
883void Compiler::error(int offset, String msg) {
ethannicholasb3058bd2016-07-01 08:22:01 -0700884 fErrorCount++;
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700885 Position pos = this->position(offset);
John Stiles8d3642e2021-01-22 09:50:04 -0500886 fErrorTextLength.push_back(fErrorText.length());
Ethan Nicholas3c729892020-12-07 12:47:17 -0500887 fErrorText += "error: " + (pos.fLine >= 1 ? to_string(pos.fLine) + ": " : "") + msg + "\n";
ethannicholasb3058bd2016-07-01 08:22:01 -0700888}
889
John Stiles8d3642e2021-01-22 09:50:04 -0500890void Compiler::setErrorCount(int c) {
891 if (c < fErrorCount) {
892 fErrorText.resize(fErrorTextLength[c]);
893 fErrorTextLength.resize(c);
894 fErrorCount = c;
895 }
896}
897
Ethan Nicholas95046142021-01-07 10:57:27 -0500898String Compiler::errorText(bool showCount) {
899 if (showCount) {
900 this->writeErrorCount();
901 }
Ethan Nicholas00543112018-07-31 09:44:36 -0400902 fErrorCount = 0;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400903 String result = fErrorText;
Ethan Nicholas95046142021-01-07 10:57:27 -0500904 fErrorText = "";
ethannicholasb3058bd2016-07-01 08:22:01 -0700905 return result;
906}
907
908void Compiler::writeErrorCount() {
909 if (fErrorCount) {
910 fErrorText += to_string(fErrorCount) + " error";
911 if (fErrorCount > 1) {
912 fErrorText += "s";
913 }
914 fErrorText += "\n";
915 }
916}
917
John Stilesa6841be2020-08-06 14:11:56 -0400918} // namespace SkSL