blob: 275eb94696fc19203557871d1070a561817a8950 [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
Ethan Nicholas55a63af2021-05-18 10:12:58 -040013#include "include/sksl/DSLCore.h"
John Stiles270cec22021-02-17 12:59:36 -050014#include "src/core/SkScopeExit.h"
Leon Scrogginsb66214e2021-02-11 17:14:18 -050015#include "src/core/SkTraceEvent.h"
John Stilesb92641c2020-08-31 18:09:01 -040016#include "src/sksl/SkSLAnalysis.h"
John Stilesf3a28db2021-03-10 23:00:47 -050017#include "src/sksl/SkSLConstantFolder.h"
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -040018#include "src/sksl/SkSLDSLParser.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050019#include "src/sksl/SkSLIRGenerator.h"
Brian Osman00185012021-02-04 16:07:11 -050020#include "src/sksl/SkSLOperators.h"
John Stiles270cec22021-02-17 12:59:36 -050021#include "src/sksl/SkSLProgramSettings.h"
Ethan Nicholasc18bb512020-07-28 14:46:53 -040022#include "src/sksl/SkSLRehydrator.h"
John Stiles3738ef52021-04-13 10:41:57 -040023#include "src/sksl/codegen/SkSLGLSLCodeGenerator.h"
John Stiles3738ef52021-04-13 10:41:57 -040024#include "src/sksl/codegen/SkSLMetalCodeGenerator.h"
25#include "src/sksl/codegen/SkSLSPIRVCodeGenerator.h"
26#include "src/sksl/codegen/SkSLSPIRVtoHLSL.h"
Ethan Nicholas55a63af2021-05-18 10:12:58 -040027#include "src/sksl/dsl/priv/DSLWriter.h"
28#include "src/sksl/dsl/priv/DSL_priv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050029#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_frag.dehydrated.sksl"
62#include "src/sksl/generated/sksl_geom.dehydrated.sksl"
63#include "src/sksl/generated/sksl_gpu.dehydrated.sksl"
Brian Osmanb06301e2020-11-06 11:45:36 -050064#include "src/sksl/generated/sksl_public.dehydrated.sksl"
John Stilesf7f36ae2021-06-08 14:06:22 -040065#include "src/sksl/generated/sksl_rt_blend.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)
John Stilesa289ac22021-05-06 07:35:35 -040086 : fCompiler(compiler) {
87 SkASSERT(!fCompiler->fSource);
Brian Osman88cda172020-10-09 12:05:16 -040088 fCompiler->fSource = source;
89 }
90
John Stilesa289ac22021-05-06 07:35:35 -040091 ~AutoSource() {
92 fCompiler->fSource = nullptr;
93 }
Brian Osman88cda172020-10-09 12:05:16 -040094
95 Compiler* fCompiler;
Brian Osman88cda172020-10-09 12:05:16 -040096};
97
John Stilesa935c3f2021-02-25 10:35:49 -050098class AutoProgramConfig {
99public:
100 AutoProgramConfig(std::shared_ptr<Context>& context, ProgramConfig* config)
101 : fContext(context.get()) {
102 SkASSERT(!fContext->fConfig);
103 fContext->fConfig = config;
104 }
105
106 ~AutoProgramConfig() {
107 fContext->fConfig = nullptr;
108 }
109
110 Context* fContext;
111};
112
John Stiles10d39d92021-05-04 16:13:14 -0400113class AutoModifiersPool {
114public:
115 AutoModifiersPool(std::shared_ptr<Context>& context, ModifiersPool* modifiersPool)
116 : fContext(context.get()) {
117 SkASSERT(!fContext->fModifiersPool);
118 fContext->fModifiersPool = modifiersPool;
119 }
120
121 ~AutoModifiersPool() {
122 fContext->fModifiersPool = nullptr;
123 }
124
125 Context* fContext;
126};
127
John Stilesd6a5f4492021-02-11 15:46:11 -0500128Compiler::Compiler(const ShaderCapsClass* caps)
John Stilesc1a98b82021-02-24 13:35:02 -0500129 : fContext(std::make_shared<Context>(/*errors=*/*this, *caps))
John Stilesa47b3512021-05-04 16:15:00 -0400130 , fInliner(fContext.get()) {
John Stilesc1a98b82021-02-24 13:35:02 -0500131 SkASSERT(caps);
John Stiles7c3515b2020-10-16 18:38:39 -0400132 fRootSymbolTable = std::make_shared<SymbolTable>(this, /*builtin=*/true);
Brian Osmanb06301e2020-11-06 11:45:36 -0500133 fPrivateSymbolTable = std::make_shared<SymbolTable>(fRootSymbolTable, /*builtin=*/true);
John Stilesc1a98b82021-02-24 13:35:02 -0500134 fIRGenerator = std::make_unique<IRGenerator>(fContext.get());
ethannicholasb3058bd2016-07-01 08:22:01 -0700135
John Stiles54e7c052021-01-11 14:22:36 -0500136#define TYPE(t) fContext->fTypes.f ## t .get()
ethannicholasb3058bd2016-07-01 08:22:01 -0700137
Brian Osmanb06301e2020-11-06 11:45:36 -0500138 const SkSL::Symbol* rootTypes[] = {
139 TYPE(Void),
Brian Salomonbf7b6202016-11-11 16:08:03 -0500140
Brian Osmanb06301e2020-11-06 11:45:36 -0500141 TYPE( Float), TYPE( Float2), TYPE( Float3), TYPE( Float4),
142 TYPE( Half), TYPE( Half2), TYPE( Half3), TYPE( Half4),
143 TYPE( Int), TYPE( Int2), TYPE( Int3), TYPE( Int4),
Brian Osmanb06301e2020-11-06 11:45:36 -0500144 TYPE( Bool), TYPE( Bool2), TYPE( Bool3), TYPE( Bool4),
Brian Salomon2a51de82016-11-16 12:06:01 -0500145
Brian Osmanc0f2b642020-12-22 13:35:55 -0500146 TYPE(Float2x2), TYPE(Float3x3), TYPE(Float4x4),
Brian Osmanc63f4312020-12-23 11:44:14 -0500147 TYPE( Half2x2), TYPE( Half3x3), TYPE(Half4x4),
Greg Daniel64773e62016-11-22 09:44:03 -0500148
Brian Osmanc63f4312020-12-23 11:44:14 -0500149 TYPE(SquareMat), TYPE(SquareHMat),
ethannicholasb3058bd2016-07-01 08:22:01 -0700150
Brian Osman20fad322020-12-23 12:42:33 -0500151 TYPE(GenType), TYPE(GenHType), TYPE(GenIType), TYPE(GenBType),
152 TYPE(Vec), TYPE(HVec), TYPE(IVec), TYPE(BVec),
Brian Osmanb06301e2020-11-06 11:45:36 -0500153
Brian Osman14d00962021-04-02 17:04:35 -0400154 TYPE(ColorFilter),
155 TYPE(Shader),
John Stilesbb2ef922021-07-26 08:32:07 -0400156 TYPE(Blender),
Brian Osmanb06301e2020-11-06 11:45:36 -0500157 };
158
159 const SkSL::Symbol* privateTypes[] = {
Brian Osman20fad322020-12-23 12:42:33 -0500160 TYPE( UInt), TYPE( UInt2), TYPE( UInt3), TYPE( UInt4),
161 TYPE( Short), TYPE( Short2), TYPE( Short3), TYPE( Short4),
162 TYPE(UShort), TYPE(UShort2), TYPE(UShort3), TYPE(UShort4),
Brian Osman20fad322020-12-23 12:42:33 -0500163
164 TYPE(GenUType), TYPE(UVec),
Ethan Nicholas722c83e2021-05-04 11:39:30 -0400165 TYPE(SVec), TYPE(USVec),
Brian Osman20fad322020-12-23 12:42:33 -0500166
Brian Osmanc0f2b642020-12-22 13:35:55 -0500167 TYPE(Float2x3), TYPE(Float2x4),
168 TYPE(Float3x2), TYPE(Float3x4),
169 TYPE(Float4x2), TYPE(Float4x3),
170
Brian Osmanc63f4312020-12-23 11:44:14 -0500171 TYPE(Half2x3), TYPE(Half2x4),
172 TYPE(Half3x2), TYPE(Half3x4),
173 TYPE(Half4x2), TYPE(Half4x3),
174
Brian Osmanc0f2b642020-12-22 13:35:55 -0500175 TYPE(Mat), TYPE(HMat),
176
Brian Osmanb06301e2020-11-06 11:45:36 -0500177 TYPE(Sampler1D), TYPE(Sampler2D), TYPE(Sampler3D),
178 TYPE(SamplerExternalOES),
Brian Osmanb06301e2020-11-06 11:45:36 -0500179 TYPE(Sampler2DRect),
Brian Osmanb06301e2020-11-06 11:45:36 -0500180
181 TYPE(ISampler2D),
Brian Osmanb06301e2020-11-06 11:45:36 -0500182 TYPE(SubpassInput), TYPE(SubpassInputMS),
183
Brian Osmanb06301e2020-11-06 11:45:36 -0500184 TYPE(Sampler),
185 TYPE(Texture2D),
186 };
187
188 for (const SkSL::Symbol* type : rootTypes) {
189 fRootSymbolTable->addWithoutOwnership(type);
190 }
191 for (const SkSL::Symbol* type : privateTypes) {
192 fPrivateSymbolTable->addWithoutOwnership(type);
193 }
194
195#undef TYPE
ethannicholasb3058bd2016-07-01 08:22:01 -0700196
Brian Osman3887a012020-09-30 13:22:27 -0400197 // sk_Caps is "builtin", but all references to it are resolved to Settings, so we don't need to
198 // treat it as builtin (ie, no need to clone it into the Program).
John Stiles10d39d92021-05-04 16:13:14 -0400199 fPrivateSymbolTable->add(std::make_unique<Variable>(/*offset=*/-1,
John Stilesa47b3512021-05-04 16:15:00 -0400200 fCoreModifiers.add(Modifiers{}),
John Stiles10d39d92021-05-04 16:13:14 -0400201 "sk_Caps",
202 fContext->fTypes.fSkCaps.get(),
203 /*builtin=*/false,
204 Variable::Storage::kGlobal));
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500205
Brian Osman3d87e9f2020-10-08 11:50:22 -0400206 fRootModule = {fRootSymbolTable, /*fIntrinsics=*/nullptr};
Brian Osmanb06301e2020-11-06 11:45:36 -0500207 fPrivateModule = {fPrivateSymbolTable, /*fIntrinsics=*/nullptr};
ethannicholasb3058bd2016-07-01 08:22:01 -0700208}
209
John Stilesdd13dba2020-10-29 10:45:34 -0400210Compiler::~Compiler() {}
ethannicholasb3058bd2016-07-01 08:22:01 -0700211
Brian Osman56269982020-11-20 12:38:07 -0500212const ParsedModule& Compiler::loadGPUModule() {
213 if (!fGPUModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500214 fGPUModule = this->parseModule(ProgramKind::kFragment, MODULE_DATA(gpu), fPrivateModule);
Brian Osman56269982020-11-20 12:38:07 -0500215 }
216 return fGPUModule;
217}
218
219const ParsedModule& Compiler::loadFragmentModule() {
220 if (!fFragmentModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500221 fFragmentModule = this->parseModule(ProgramKind::kFragment, MODULE_DATA(frag),
Brian Osman56269982020-11-20 12:38:07 -0500222 this->loadGPUModule());
223 }
224 return fFragmentModule;
225}
226
227const ParsedModule& Compiler::loadVertexModule() {
228 if (!fVertexModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500229 fVertexModule = this->parseModule(ProgramKind::kVertex, MODULE_DATA(vert),
Brian Osman56269982020-11-20 12:38:07 -0500230 this->loadGPUModule());
231 }
232 return fVertexModule;
233}
234
Brian Osman88cda172020-10-09 12:05:16 -0400235const ParsedModule& Compiler::loadGeometryModule() {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400236 if (!fGeometryModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500237 fGeometryModule = this->parseModule(ProgramKind::kGeometry, MODULE_DATA(geom),
Brian Osman56269982020-11-20 12:38:07 -0500238 this->loadGPUModule());
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400239 }
Brian Osman88cda172020-10-09 12:05:16 -0400240 return fGeometryModule;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400241}
242
Brian Osmanb06301e2020-11-06 11:45:36 -0500243const ParsedModule& Compiler::loadPublicModule() {
244 if (!fPublicModule.fSymbols) {
John Stilesdbd4e6f2021-02-16 13:29:15 -0500245 fPublicModule = this->parseModule(ProgramKind::kGeneric, MODULE_DATA(public), fRootModule);
Brian Osmanb06301e2020-11-06 11:45:36 -0500246 }
247 return fPublicModule;
248}
249
Brian Osmancbb60bd2021-04-12 09:49:20 -0400250static void add_glsl_type_aliases(SkSL::SymbolTable* symbols, const SkSL::BuiltinTypes& types) {
251 // Add some aliases to the runtime effect modules so that it's friendlier, and more like GLSL
252 symbols->addAlias("vec2", types.fFloat2.get());
253 symbols->addAlias("vec3", types.fFloat3.get());
254 symbols->addAlias("vec4", types.fFloat4.get());
255
256 symbols->addAlias("ivec2", types.fInt2.get());
257 symbols->addAlias("ivec3", types.fInt3.get());
258 symbols->addAlias("ivec4", types.fInt4.get());
259
260 symbols->addAlias("bvec2", types.fBool2.get());
261 symbols->addAlias("bvec3", types.fBool3.get());
262 symbols->addAlias("bvec4", types.fBool4.get());
263
264 symbols->addAlias("mat2", types.fFloat2x2.get());
265 symbols->addAlias("mat3", types.fFloat3x3.get());
266 symbols->addAlias("mat4", types.fFloat4x4.get());
267}
268
Brian Osmancbb60bd2021-04-12 09:49:20 -0400269const ParsedModule& Compiler::loadRuntimeColorFilterModule() {
270 if (!fRuntimeColorFilterModule.fSymbols) {
271 fRuntimeColorFilterModule = this->parseModule(ProgramKind::kRuntimeColorFilter,
272 MODULE_DATA(rt_colorfilter),
273 this->loadPublicModule());
274 add_glsl_type_aliases(fRuntimeColorFilterModule.fSymbols.get(), fContext->fTypes);
275 }
276 return fRuntimeColorFilterModule;
277}
278
279const ParsedModule& Compiler::loadRuntimeShaderModule() {
280 if (!fRuntimeShaderModule.fSymbols) {
281 fRuntimeShaderModule = this->parseModule(
282 ProgramKind::kRuntimeShader, MODULE_DATA(rt_shader), this->loadPublicModule());
283 add_glsl_type_aliases(fRuntimeShaderModule.fSymbols.get(), fContext->fTypes);
284 }
285 return fRuntimeShaderModule;
286}
287
John Stiles2d8b8352021-06-16 11:33:13 -0400288const ParsedModule& Compiler::loadRuntimeBlenderModule() {
289 if (!fRuntimeBlenderModule.fSymbols) {
290 fRuntimeBlenderModule = this->parseModule(
291 ProgramKind::kRuntimeBlender, MODULE_DATA(rt_blend), this->loadPublicModule());
292 add_glsl_type_aliases(fRuntimeBlenderModule.fSymbols.get(), fContext->fTypes);
John Stilesf7f36ae2021-06-08 14:06:22 -0400293 }
John Stiles2d8b8352021-06-16 11:33:13 -0400294 return fRuntimeBlenderModule;
John Stilesf7f36ae2021-06-08 14:06:22 -0400295}
296
John Stilesdbd4e6f2021-02-16 13:29:15 -0500297const ParsedModule& Compiler::moduleForProgramKind(ProgramKind kind) {
Brian Osman88cda172020-10-09 12:05:16 -0400298 switch (kind) {
Brian Osmancbb60bd2021-04-12 09:49:20 -0400299 case ProgramKind::kVertex: return this->loadVertexModule(); break;
300 case ProgramKind::kFragment: return this->loadFragmentModule(); break;
301 case ProgramKind::kGeometry: return this->loadGeometryModule(); break;
Brian Osmancbb60bd2021-04-12 09:49:20 -0400302 case ProgramKind::kRuntimeColorFilter: return this->loadRuntimeColorFilterModule(); break;
303 case ProgramKind::kRuntimeShader: return this->loadRuntimeShaderModule(); break;
John Stiles2d8b8352021-06-16 11:33:13 -0400304 case ProgramKind::kRuntimeBlender: return this->loadRuntimeBlenderModule(); break;
Brian Osmancbb60bd2021-04-12 09:49:20 -0400305 case ProgramKind::kGeneric: return this->loadPublicModule(); break;
Brian Osman88cda172020-10-09 12:05:16 -0400306 }
307 SkUNREACHABLE;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400308}
309
John Stilesdbd4e6f2021-02-16 13:29:15 -0500310LoadedModule Compiler::loadModule(ProgramKind kind,
Brian Osman3d87e9f2020-10-08 11:50:22 -0400311 ModuleData data,
John Stilesa935c3f2021-02-25 10:35:49 -0500312 std::shared_ptr<SymbolTable> base,
313 bool dehydrate) {
314 if (dehydrate) {
315 // NOTE: This is a workaround. When dehydrating includes, skslc doesn't know which module
316 // it's preparing, nor what the correct base module is. We can't use 'Root', because many
317 // GPU intrinsics reference private types, like samplers or textures. Today, 'Private' does
318 // contain the union of all known types, so this is safe. If we ever have types that only
319 // exist in 'Public' (for example), this logic needs to be smarter (by choosing the correct
320 // base for the module we're compiling).
Brian Osmanb06301e2020-11-06 11:45:36 -0500321 base = fPrivateSymbolTable;
Brian Osman3d87e9f2020-10-08 11:50:22 -0400322 }
John Stilesa935c3f2021-02-25 10:35:49 -0500323 SkASSERT(base);
324
John Stilesa47b3512021-05-04 16:15:00 -0400325 // Put the core-module modifier pool into the context.
326 AutoModifiersPool autoPool(fContext, &fCoreModifiers);
John Stiles10d39d92021-05-04 16:13:14 -0400327
John Stilesa935c3f2021-02-25 10:35:49 -0500328 // Built-in modules always use default program settings.
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400329 Program::Settings settings;
330 settings.fReplaceSettings = !dehydrate;
Brian Osman3d87e9f2020-10-08 11:50:22 -0400331
332#if defined(SKSL_STANDALONE)
333 SkASSERT(data.fPath);
334 std::ifstream in(data.fPath);
John Stilesd51c9792021-03-18 11:40:14 -0400335 String text{std::istreambuf_iterator<char>(in), std::istreambuf_iterator<char>()};
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400336 if (in.rdstate()) {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400337 printf("error reading %s\n", data.fPath);
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400338 abort();
339 }
Brian Osmane498b3c2020-09-23 14:42:11 -0400340 const String* source = fRootSymbolTable->takeOwnershipOfString(std::move(text));
John Stilesd1204642021-02-17 16:30:02 -0500341
Brian Osman88cda172020-10-09 12:05:16 -0400342 ParsedModule baseModule = {base, /*fIntrinsics=*/nullptr};
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400343 std::vector<std::unique_ptr<ProgramElement>> elements;
344 std::vector<const ProgramElement*> sharedElements;
345 dsl::StartModule(this, kind, settings, baseModule);
346 AutoSource as(this, source);
John Stilesd1204642021-02-17 16:30:02 -0500347 IRGenerator::IRBundle ir = fIRGenerator->convertProgram(baseModule, /*isBuiltinCode=*/true,
Ethan Nicholas6823b502021-06-15 11:42:07 -0400348 *source);
Brian Osman133724c2020-10-28 14:14:39 -0400349 SkASSERT(ir.fSharedElements.empty());
Brian Osman0006ad02020-11-18 15:38:39 -0500350 LoadedModule module = { kind, std::move(ir.fSymbolTable), std::move(ir.fElements) };
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400351 dsl::End();
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400352 if (this->fErrorCount) {
353 printf("Unexpected errors: %s\n", this->fErrorText.c_str());
Brian Osman3d87e9f2020-10-08 11:50:22 -0400354 SkDEBUGFAILF("%s %s\n", data.fPath, this->fErrorText.c_str());
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400355 }
Brian Osman3d87e9f2020-10-08 11:50:22 -0400356#else
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400357 ProgramConfig config;
358 config.fKind = kind;
359 config.fSettings = settings;
360 AutoProgramConfig autoConfig(fContext, &config);
Brian Osman3d87e9f2020-10-08 11:50:22 -0400361 SkASSERT(data.fData && (data.fSize != 0));
John Stiles10d39d92021-05-04 16:13:14 -0400362 Rehydrator rehydrator(fContext.get(), base, data.fData, data.fSize);
Brian Osman0006ad02020-11-18 15:38:39 -0500363 LoadedModule module = { kind, rehydrator.symbolTable(), rehydrator.elements() };
Brian Osman3d87e9f2020-10-08 11:50:22 -0400364#endif
365
366 return module;
367}
368
John Stilesdbd4e6f2021-02-16 13:29:15 -0500369ParsedModule Compiler::parseModule(ProgramKind kind, ModuleData data, const ParsedModule& base) {
John Stilesa935c3f2021-02-25 10:35:49 -0500370 LoadedModule module = this->loadModule(kind, data, base.fSymbols, /*dehydrate=*/false);
Brian Osman0006ad02020-11-18 15:38:39 -0500371 this->optimize(module);
Brian Osman3d87e9f2020-10-08 11:50:22 -0400372
373 // For modules that just declare (but don't define) intrinsic functions, there will be no new
374 // program elements. In that case, we can share our parent's intrinsic map:
Brian Osman0006ad02020-11-18 15:38:39 -0500375 if (module.fElements.empty()) {
John Stiles10d39d92021-05-04 16:13:14 -0400376 return ParsedModule{module.fSymbols, base.fIntrinsics};
Brian Osman3d87e9f2020-10-08 11:50:22 -0400377 }
378
379 auto intrinsics = std::make_shared<IRIntrinsicMap>(base.fIntrinsics.get());
380
381 // Now, transfer all of the program elements to an intrinsic map. This maps certain types of
382 // global objects to the declaring ProgramElement.
Brian Osman0006ad02020-11-18 15:38:39 -0500383 for (std::unique_ptr<ProgramElement>& element : module.fElements) {
Brian Osman3d87e9f2020-10-08 11:50:22 -0400384 switch (element->kind()) {
385 case ProgramElement::Kind::kFunction: {
386 const FunctionDefinition& f = element->as<FunctionDefinition>();
Ethan Nicholas0a5d0962020-10-14 13:33:18 -0400387 SkASSERT(f.declaration().isBuiltin());
388 intrinsics->insertOrDie(f.declaration().description(), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400389 break;
390 }
John Stiles569249b2020-11-03 12:18:22 -0500391 case ProgramElement::Kind::kFunctionPrototype: {
392 // These are already in the symbol table.
393 break;
394 }
Brian Osman3d87e9f2020-10-08 11:50:22 -0400395 case ProgramElement::Kind::kGlobalVar: {
Ethan Nicholasc51f33e2020-10-13 13:49:44 -0400396 const GlobalVarDeclaration& global = element->as<GlobalVarDeclaration>();
397 const Variable& var = global.declaration()->as<VarDeclaration>().var();
398 SkASSERT(var.isBuiltin());
Ethan Nicholasd2e09602021-06-10 11:21:59 -0400399 intrinsics->insertOrDie(String(var.name()), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400400 break;
401 }
402 case ProgramElement::Kind::kInterfaceBlock: {
Ethan Nicholaseaf47882020-10-15 10:10:08 -0400403 const Variable& var = element->as<InterfaceBlock>().variable();
404 SkASSERT(var.isBuiltin());
Ethan Nicholasd2e09602021-06-10 11:21:59 -0400405 intrinsics->insertOrDie(String(var.name()), std::move(element));
Brian Osman3d87e9f2020-10-08 11:50:22 -0400406 break;
407 }
408 default:
409 printf("Unsupported element: %s\n", element->description().c_str());
410 SkASSERT(false);
411 break;
412 }
413 }
414
John Stiles10d39d92021-05-04 16:13:14 -0400415 return ParsedModule{module.fSymbols, std::move(intrinsics)};
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400416}
417
Brian Osman32d53552020-09-23 13:55:20 -0400418std::unique_ptr<Program> Compiler::convertProgram(
John Stilesdbd4e6f2021-02-16 13:29:15 -0500419 ProgramKind kind,
Brian Osman32d53552020-09-23 13:55:20 -0400420 String text,
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400421 Program::Settings settings) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400422 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::convertProgram");
Leon Scrogginsb66214e2021-02-11 17:14:18 -0500423
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400424 SkASSERT(!settings.fExternalFunctions || (kind == ProgramKind::kGeneric));
Ethan Nicholas91164d12019-05-15 15:29:54 -0400425
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400426#if !SKSL_DSL_PARSER
Brian Osman0006ad02020-11-18 15:38:39 -0500427 // Loading and optimizing our base module might reset the inliner, so do that first,
428 // *then* configure the inliner with the settings for this program.
429 const ParsedModule& baseModule = this->moduleForProgramKind(kind);
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400430#endif
Brian Osman0006ad02020-11-18 15:38:39 -0500431
John Stiles2ee4d7a2021-03-30 10:30:47 -0400432 // Honor our optimization-override flags.
433 switch (sOptimizer) {
434 case OverrideFlag::kDefault:
435 break;
436 case OverrideFlag::kOff:
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400437 settings.fOptimize = false;
John Stiles2ee4d7a2021-03-30 10:30:47 -0400438 break;
439 case OverrideFlag::kOn:
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400440 settings.fOptimize = true;
John Stiles2ee4d7a2021-03-30 10:30:47 -0400441 break;
442 }
443
444 switch (sInliner) {
445 case OverrideFlag::kDefault:
446 break;
447 case OverrideFlag::kOff:
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400448 settings.fInlineThreshold = 0;
John Stiles2ee4d7a2021-03-30 10:30:47 -0400449 break;
450 case OverrideFlag::kOn:
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400451 if (settings.fInlineThreshold == 0) {
452 settings.fInlineThreshold = kDefaultInlineThreshold;
John Stiles2ee4d7a2021-03-30 10:30:47 -0400453 }
454 break;
455 }
John Stiles7247b482021-03-08 10:40:35 -0500456
457 // Disable optimization settings that depend on a parent setting which has been disabled.
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400458 settings.fInlineThreshold *= (int)settings.fOptimize;
459 settings.fRemoveDeadFunctions &= settings.fOptimize;
460 settings.fRemoveDeadVariables &= settings.fOptimize;
John Stiles7247b482021-03-08 10:40:35 -0500461
ethannicholasb3058bd2016-07-01 08:22:01 -0700462 fErrorText = "";
463 fErrorCount = 0;
John Stiles10d39d92021-05-04 16:13:14 -0400464 fInliner.reset();
Brian Osman88cda172020-10-09 12:05:16 -0400465
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400466#if SKSL_DSL_PARSER
467 settings.fDSLMangling = false;
468 return DSLParser(this, settings, kind, text).program();
469#else
John Stiles10d39d92021-05-04 16:13:14 -0400470 auto textPtr = std::make_unique<String>(std::move(text));
John Stilesa289ac22021-05-06 07:35:35 -0400471 AutoSource as(this, textPtr.get());
Brian Osman88cda172020-10-09 12:05:16 -0400472
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400473 dsl::Start(this, kind, settings);
John Stilesd1204642021-02-17 16:30:02 -0500474 IRGenerator::IRBundle ir = fIRGenerator->convertProgram(baseModule, /*isBuiltinCode=*/false,
Ethan Nicholas6823b502021-06-15 11:42:07 -0400475 *textPtr);
Ethan Nicholas4f3e6a22021-06-15 09:17:05 -0400476 // Ideally, we would just use dsl::ReleaseProgram and not have to do any manual mucking about
477 // with the memory pool, but we've got some impedance mismatches to solve first
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400478 Pool* memoryPool = dsl::DSLWriter::MemoryPool().get();
John Stiles270cec22021-02-17 12:59:36 -0500479 auto program = std::make_unique<Program>(std::move(textPtr),
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400480 std::move(dsl::DSLWriter::GetProgramConfig()),
John Stiles5c7bb322020-10-22 11:09:15 -0400481 fContext,
482 std::move(ir.fElements),
Brian Osman133724c2020-10-28 14:14:39 -0400483 std::move(ir.fSharedElements),
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400484 std::move(dsl::DSLWriter::GetModifiersPool()),
John Stiles5c7bb322020-10-22 11:09:15 -0400485 std::move(ir.fSymbolTable),
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400486 std::move(dsl::DSLWriter::MemoryPool()),
John Stiles5c7bb322020-10-22 11:09:15 -0400487 ir.fInputs);
488 bool success = false;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500489 if (fErrorCount) {
John Stiles5c7bb322020-10-22 11:09:15 -0400490 // Do not return programs that failed to compile.
John Stiles7247b482021-03-08 10:40:35 -0500491 } else if (!this->optimize(*program)) {
John Stiles5c7bb322020-10-22 11:09:15 -0400492 // Do not return programs that failed to optimize.
493 } else {
494 // We have a successful program!
495 success = true;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500496 }
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400497 dsl::End();
498 if (memoryPool) {
499 memoryPool->detachFromThread();
Brian Osman28f702c2021-02-02 11:52:07 -0500500 }
John Stiles5c7bb322020-10-22 11:09:15 -0400501 return success ? std::move(program) : nullptr;
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400502#endif // SKSL_DSL_PARSER
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500503}
504
John Stilesbb1505f2021-02-12 09:17:53 -0500505void Compiler::verifyStaticTests(const Program& program) {
506 class StaticTestVerifier : public ProgramVisitor {
507 public:
508 StaticTestVerifier(ErrorReporter* r) : fReporter(r) {}
509
510 using ProgramVisitor::visitProgramElement;
511
512 bool visitStatement(const Statement& stmt) override {
513 switch (stmt.kind()) {
514 case Statement::Kind::kIf:
515 if (stmt.as<IfStatement>().isStatic()) {
516 fReporter->error(stmt.fOffset, "static if has non-static test");
517 }
518 break;
519
520 case Statement::Kind::kSwitch:
521 if (stmt.as<SwitchStatement>().isStatic()) {
522 fReporter->error(stmt.fOffset, "static switch has non-static test");
523 }
524 break;
525
526 default:
527 break;
528 }
529 return INHERITED::visitStatement(stmt);
530 }
531
John Stiles59e34562021-02-12 16:56:39 -0500532 bool visitExpression(const Expression&) override {
533 // We aren't looking for anything inside an Expression, so skip them entirely.
534 return false;
535 }
536
John Stilesbb1505f2021-02-12 09:17:53 -0500537 private:
538 using INHERITED = ProgramVisitor;
539 ErrorReporter* fReporter;
540 };
541
542 // If invalid static tests are permitted, we don't need to check anything.
John Stilesd1204642021-02-17 16:30:02 -0500543 if (fContext->fConfig->fSettings.fPermitInvalidStaticTests) {
John Stilesbb1505f2021-02-12 09:17:53 -0500544 return;
545 }
546
547 // Check all of the program's owned elements. (Built-in elements are assumed to be valid.)
548 StaticTestVerifier visitor{this};
549 for (const std::unique_ptr<ProgramElement>& element : program.ownedElements()) {
550 if (element->is<FunctionDefinition>()) {
551 visitor.visitProgramElement(*element);
552 }
553 }
554}
555
Brian Osman0006ad02020-11-18 15:38:39 -0500556bool Compiler::optimize(LoadedModule& module) {
557 SkASSERT(!fErrorCount);
Brian Osman0006ad02020-11-18 15:38:39 -0500558
John Stiles270cec22021-02-17 12:59:36 -0500559 // Create a temporary program configuration with default settings.
560 ProgramConfig config;
561 config.fKind = module.fKind;
John Stilesa935c3f2021-02-25 10:35:49 -0500562 AutoProgramConfig autoConfig(fContext, &config);
John Stiles270cec22021-02-17 12:59:36 -0500563
John Stilesd1204642021-02-17 16:30:02 -0500564 // Reset the Inliner.
John Stiles10d39d92021-05-04 16:13:14 -0400565 fInliner.reset();
John Stiles270cec22021-02-17 12:59:36 -0500566
567 std::unique_ptr<ProgramUsage> usage = Analysis::GetUsage(module);
Brian Osman0006ad02020-11-18 15:38:39 -0500568
569 while (fErrorCount == 0) {
Brian Osman0006ad02020-11-18 15:38:39 -0500570 // Perform inline-candidate analysis and inline any functions deemed suitable.
John Stilesf3a28db2021-03-10 23:00:47 -0500571 if (!fInliner.analyze(module.fElements, module.fSymbols, usage.get())) {
Brian Osman0006ad02020-11-18 15:38:39 -0500572 break;
573 }
574 }
575 return fErrorCount == 0;
576}
577
John Stiles0bfeae62021-03-11 09:09:42 -0500578bool Compiler::removeDeadFunctions(Program& program, ProgramUsage* usage) {
579 bool madeChanges = false;
580
581 if (program.fConfig->fSettings.fRemoveDeadFunctions) {
582 auto isDeadFunction = [&](const ProgramElement* element) {
583 if (!element->is<FunctionDefinition>()) {
584 return false;
585 }
586 const FunctionDefinition& fn = element->as<FunctionDefinition>();
John Stilese8da4d22021-03-24 09:19:45 -0400587 if (fn.declaration().isMain() || usage->get(fn.declaration()) > 0) {
John Stiles0bfeae62021-03-11 09:09:42 -0500588 return false;
589 }
590 usage->remove(*element);
591 madeChanges = true;
592 return true;
593 };
594
595 program.fElements.erase(std::remove_if(program.fElements.begin(),
596 program.fElements.end(),
597 [&](const std::unique_ptr<ProgramElement>& element) {
598 return isDeadFunction(element.get());
599 }),
600 program.fElements.end());
601 program.fSharedElements.erase(std::remove_if(program.fSharedElements.begin(),
602 program.fSharedElements.end(),
603 isDeadFunction),
604 program.fSharedElements.end());
605 }
606 return madeChanges;
607}
608
609bool Compiler::removeDeadGlobalVariables(Program& program, ProgramUsage* usage) {
610 bool madeChanges = false;
611
612 if (program.fConfig->fSettings.fRemoveDeadVariables) {
613 auto isDeadVariable = [&](const ProgramElement* element) {
614 if (!element->is<GlobalVarDeclaration>()) {
615 return false;
616 }
617 const GlobalVarDeclaration& global = element->as<GlobalVarDeclaration>();
618 const VarDeclaration& varDecl = global.declaration()->as<VarDeclaration>();
619 if (!usage->isDead(varDecl.var())) {
620 return false;
621 }
622 madeChanges = true;
623 return true;
624 };
625
626 program.fElements.erase(std::remove_if(program.fElements.begin(),
627 program.fElements.end(),
628 [&](const std::unique_ptr<ProgramElement>& element) {
629 return isDeadVariable(element.get());
630 }),
631 program.fElements.end());
632 program.fSharedElements.erase(std::remove_if(program.fSharedElements.begin(),
633 program.fSharedElements.end(),
634 isDeadVariable),
635 program.fSharedElements.end());
636 }
637 return madeChanges;
638}
639
John Stiles26541872021-03-16 12:19:54 -0400640bool Compiler::removeDeadLocalVariables(Program& program, ProgramUsage* usage) {
641 class DeadLocalVariableEliminator : public ProgramWriter {
642 public:
643 DeadLocalVariableEliminator(const Context& context, ProgramUsage* usage)
644 : fContext(context)
645 , fUsage(usage) {}
646
647 using ProgramWriter::visitProgramElement;
648
649 bool visitExpressionPtr(std::unique_ptr<Expression>& expr) override {
650 // We don't need to look inside expressions at all.
651 return false;
652 }
653
654 bool visitStatementPtr(std::unique_ptr<Statement>& stmt) override {
655 if (stmt->is<VarDeclaration>()) {
656 VarDeclaration& varDecl = stmt->as<VarDeclaration>();
657 const Variable* var = &varDecl.var();
658 ProgramUsage::VariableCounts* counts = fUsage->fVariableCounts.find(var);
659 SkASSERT(counts);
660 SkASSERT(counts->fDeclared);
661 if (CanEliminate(var, *counts)) {
662 if (var->initialValue()) {
663 // The variable has an initial-value expression, which might have side
664 // effects. ExpressionStatement::Make will preserve side effects, but
665 // replaces pure expressions with Nop.
666 fUsage->remove(stmt.get());
667 stmt = ExpressionStatement::Make(fContext, std::move(varDecl.value()));
668 fUsage->add(stmt.get());
669 } else {
670 // The variable has no initial-value and can be cleanly eliminated.
671 fUsage->remove(stmt.get());
672 stmt = std::make_unique<Nop>();
673 }
674 fMadeChanges = true;
675 }
676 return false;
677 }
678 return INHERITED::visitStatementPtr(stmt);
679 }
680
681 static bool CanEliminate(const Variable* var, const ProgramUsage::VariableCounts& counts) {
682 if (!counts.fDeclared || counts.fRead || var->storage() != VariableStorage::kLocal) {
683 return false;
684 }
685 if (var->initialValue()) {
686 SkASSERT(counts.fWrite >= 1);
687 return counts.fWrite == 1;
688 } else {
689 return counts.fWrite == 0;
690 }
691 }
692
693 bool fMadeChanges = false;
694 const Context& fContext;
695 ProgramUsage* fUsage;
696
697 using INHERITED = ProgramWriter;
698 };
699
700 DeadLocalVariableEliminator visitor{*fContext, usage};
701
702 if (program.fConfig->fSettings.fRemoveDeadVariables) {
703 for (auto& [var, counts] : usage->fVariableCounts) {
704 if (DeadLocalVariableEliminator::CanEliminate(var, counts)) {
705 // This program contains at least one dead local variable.
706 // Scan the program for any dead local variables and eliminate them all.
707 for (std::unique_ptr<ProgramElement>& pe : program.ownedElements()) {
708 if (pe->is<FunctionDefinition>()) {
709 visitor.visitProgramElement(*pe);
710 }
711 }
712 break;
713 }
714 }
715 }
716
717 return visitor.fMadeChanges;
718}
719
John Stiles25be58e2021-05-20 14:38:40 -0400720void Compiler::removeUnreachableCode(Program& program, ProgramUsage* usage) {
721 class UnreachableCodeEliminator : public ProgramWriter {
722 public:
723 UnreachableCodeEliminator(const Context& context, ProgramUsage* usage)
724 : fContext(context)
725 , fUsage(usage) {
726 fFoundFunctionExit.push(false);
727 fFoundLoopExit.push(false);
728 }
729
730 using ProgramWriter::visitProgramElement;
731
732 bool visitExpressionPtr(std::unique_ptr<Expression>& expr) override {
733 // We don't need to look inside expressions at all.
734 return false;
735 }
736
737 bool visitStatementPtr(std::unique_ptr<Statement>& stmt) override {
738 if (fFoundFunctionExit.top() || fFoundLoopExit.top()) {
739 // If we already found an exit in this section, anything beyond it is dead code.
740 if (!stmt->is<Nop>()) {
741 // Eliminate the dead statement by substituting a Nop.
742 fUsage->remove(stmt.get());
743 stmt = std::make_unique<Nop>();
744 }
745 return false;
746 }
747
748 switch (stmt->kind()) {
749 case Statement::Kind::kReturn:
750 case Statement::Kind::kDiscard:
751 // We found a function exit on this path.
752 fFoundFunctionExit.top() = true;
753 break;
754
755 case Statement::Kind::kBreak:
756 case Statement::Kind::kContinue:
757 // We found a loop exit on this path. Note that we skip over switch statements
758 // completely when eliminating code, so any `break` statement would be breaking
759 // out of a loop, not out of a switch.
760 fFoundLoopExit.top() = true;
761 break;
762
763 case Statement::Kind::kExpression:
764 case Statement::Kind::kInlineMarker:
765 case Statement::Kind::kNop:
766 case Statement::Kind::kVarDeclaration:
767 // These statements don't affect control flow.
768 break;
769
770 case Statement::Kind::kBlock:
771 // Blocks are on the straight-line path and don't affect control flow.
772 return INHERITED::visitStatementPtr(stmt);
773
774 case Statement::Kind::kDo: {
775 // Function-exits are allowed to propagate outside of a do-loop, because it
776 // always executes its body at least once.
777 fFoundLoopExit.push(false);
778 bool result = INHERITED::visitStatementPtr(stmt);
779 fFoundLoopExit.pop();
780 return result;
781 }
782 case Statement::Kind::kFor: {
783 // Function-exits are not allowed to propagate out, because a for-loop or while-
784 // loop could potentially run zero times.
785 fFoundFunctionExit.push(false);
786 fFoundLoopExit.push(false);
787 bool result = INHERITED::visitStatementPtr(stmt);
788 fFoundLoopExit.pop();
789 fFoundFunctionExit.pop();
790 return result;
791 }
792 case Statement::Kind::kIf: {
793 // This statement is conditional and encloses two inner sections of code.
794 // If both sides contain a function-exit or loop-exit, that exit is allowed to
795 // propagate out.
796 IfStatement& ifStmt = stmt->as<IfStatement>();
797
798 fFoundFunctionExit.push(false);
799 fFoundLoopExit.push(false);
800 bool result = (ifStmt.ifTrue() && this->visitStatementPtr(ifStmt.ifTrue()));
801 bool foundFunctionExitOnTrue = fFoundFunctionExit.top();
802 bool foundLoopExitOnTrue = fFoundLoopExit.top();
803 fFoundFunctionExit.pop();
804 fFoundLoopExit.pop();
805
806 fFoundFunctionExit.push(false);
807 fFoundLoopExit.push(false);
808 result |= (ifStmt.ifFalse() && this->visitStatementPtr(ifStmt.ifFalse()));
809 bool foundFunctionExitOnFalse = fFoundFunctionExit.top();
810 bool foundLoopExitOnFalse = fFoundLoopExit.top();
811 fFoundFunctionExit.pop();
812 fFoundLoopExit.pop();
813
814 fFoundFunctionExit.top() |= foundFunctionExitOnTrue && foundFunctionExitOnFalse;
815 fFoundLoopExit.top() |= foundLoopExitOnTrue && foundLoopExitOnFalse;
816 return result;
817 }
818 case Statement::Kind::kSwitch:
819 case Statement::Kind::kSwitchCase:
820 // We skip past switch statements entirely when scanning for dead code. Their
821 // control flow is quite complex and we already do a good job of flattening out
822 // switches on constant values.
823 break;
824 }
825
826 return false;
827 }
828
829 const Context& fContext;
830 ProgramUsage* fUsage;
831 std::stack<bool> fFoundFunctionExit;
832 std::stack<bool> fFoundLoopExit;
833
834 using INHERITED = ProgramWriter;
835 };
836
837 for (std::unique_ptr<ProgramElement>& pe : program.ownedElements()) {
838 if (pe->is<FunctionDefinition>()) {
839 UnreachableCodeEliminator visitor{*fContext, usage};
840 visitor.visitProgramElement(*pe);
841 }
842 }
843}
844
Ethan Nicholas00543112018-07-31 09:44:36 -0400845bool Compiler::optimize(Program& program) {
John Stiles7247b482021-03-08 10:40:35 -0500846 // The optimizer only needs to run when it is enabled.
847 if (!program.fConfig->fSettings.fOptimize) {
848 return true;
849 }
850
Ethan Nicholas00543112018-07-31 09:44:36 -0400851 SkASSERT(!fErrorCount);
Brian Osman010ce6a2020-10-19 16:34:10 -0400852 ProgramUsage* usage = program.fUsage.get();
John Stiles7954d6c2020-09-01 10:53:02 -0400853
John Stilesb6664582021-03-19 09:46:00 -0400854 if (fErrorCount == 0) {
John Stiles87fc6572021-04-01 14:56:34 +0000855 // Run the inliner only once; it is expensive! Multiple passes can occasionally shake out
856 // more wins, but it's diminishing returns.
857 fInliner.analyze(program.ownedElements(), program.fSymbols, usage);
Ethan Nicholas34b19c52020-09-14 11:33:47 -0400858
John Stilesb6664582021-03-19 09:46:00 -0400859 while (this->removeDeadFunctions(program, usage)) {
860 // Removing dead functions may cause more functions to become unreferenced. Try again.
Ethan Nicholas34b19c52020-09-14 11:33:47 -0400861 }
John Stilesb6664582021-03-19 09:46:00 -0400862 while (this->removeDeadLocalVariables(program, usage)) {
863 // Removing dead variables may cause more variables to become unreferenced. Try again.
864 }
John Stiles25be58e2021-05-20 14:38:40 -0400865 // Unreachable code can confuse some drivers, so it's worth removing. (skia:12012)
866 this->removeUnreachableCode(program, usage);
867
Brian Osman8c264792021-07-01 16:41:27 -0400868 this->removeDeadGlobalVariables(program, usage);
Ethan Nicholas00543112018-07-31 09:44:36 -0400869 }
John Stilesbb1505f2021-02-12 09:17:53 -0500870
871 if (fErrorCount == 0) {
872 this->verifyStaticTests(program);
873 }
874
Ethan Nicholas00543112018-07-31 09:44:36 -0400875 return fErrorCount == 0;
876}
877
Brian Osmanfb32ddf2019-06-18 10:14:20 -0400878#if defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
879
Ethan Nicholas00543112018-07-31 09:44:36 -0400880bool Compiler::toSPIRV(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400881 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toSPIRV");
John Stilesa289ac22021-05-06 07:35:35 -0400882 AutoSource as(this, program.fSource.get());
Brian Salomond8d85b92021-07-07 09:41:17 -0400883 ProgramSettings settings;
884 settings.fDSLUseMemoryPool = false;
885 dsl::Start(this, program.fConfig->fKind, settings);
886 dsl::DSLWriter::IRGenerator().fSymbolTable = program.fSymbols;
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400887#ifdef SK_ENABLE_SPIRV_VALIDATION
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400888 StringStream buffer;
Brian Osman8b43dad2020-10-09 13:31:42 -0400889 SPIRVCodeGenerator cg(fContext.get(), &program, this, &buffer);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400890 bool result = cg.generateCode();
John Stiles270cec22021-02-17 12:59:36 -0500891 if (result && program.fConfig->fSettings.fValidateSPIRV) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400892 spvtools::SpirvTools tools(SPV_ENV_VULKAN_1_0);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400893 const String& data = buffer.str();
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400894 SkASSERT(0 == data.size() % 4);
Brian Osman8d09d4a2020-11-24 15:51:06 -0500895 String errors;
896 auto dumpmsg = [&errors](spv_message_level_t, const char*, const spv_position_t&,
897 const char* m) {
898 errors.appendf("SPIR-V validation error: %s\n", m);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400899 };
900 tools.SetMessageConsumer(dumpmsg);
Brian Osman8d09d4a2020-11-24 15:51:06 -0500901
902 // Verify that the SPIR-V we produced is valid. At runtime, we will abort() with a message
903 // explaining the error. In standalone mode (skslc), we will send the message, plus the
904 // entire disassembled SPIR-V (for easier context & debugging) as *our* error message.
905 result = tools.Validate((const uint32_t*) data.c_str(), data.size() / 4);
906
907 if (!result) {
908#if defined(SKSL_STANDALONE)
909 // Convert the string-stream to a SPIR-V disassembly.
910 std::string disassembly;
911 if (tools.Disassemble((const uint32_t*)data.data(), data.size() / 4, &disassembly)) {
912 errors.append(disassembly);
913 }
914 this->error(-1, errors);
915#else
916 SkDEBUGFAILF("%s", errors.c_str());
917#endif
918 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400919 out.write(data.c_str(), data.size());
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400920 }
921#else
Brian Osman8b43dad2020-10-09 13:31:42 -0400922 SPIRVCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500923 bool result = cg.generateCode();
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400924#endif
Brian Salomond8d85b92021-07-07 09:41:17 -0400925 dsl::End();
Ethan Nicholasce33f102016-12-09 17:22:59 -0500926 return result;
927}
928
Ethan Nicholas00543112018-07-31 09:44:36 -0400929bool Compiler::toSPIRV(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400930 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500931 bool result = this->toSPIRV(program, buffer);
932 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400933 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500934 }
935 return result;
936}
937
Ethan Nicholas00543112018-07-31 09:44:36 -0400938bool Compiler::toGLSL(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400939 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toGLSL");
Brian Osman88cda172020-10-09 12:05:16 -0400940 AutoSource as(this, program.fSource.get());
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400941 GLSLCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500942 bool result = cg.generateCode();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500943 return result;
944}
945
Ethan Nicholas00543112018-07-31 09:44:36 -0400946bool Compiler::toGLSL(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400947 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500948 bool result = this->toGLSL(program, buffer);
949 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400950 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500951 }
952 return result;
953}
954
Brian Osmanc0243912020-02-19 15:35:26 -0500955bool Compiler::toHLSL(Program& program, String* out) {
956 String spirv;
957 if (!this->toSPIRV(program, &spirv)) {
958 return false;
959 }
960
961 return SPIRVtoHLSL(spirv, out);
962}
963
Ethan Nicholas00543112018-07-31 09:44:36 -0400964bool Compiler::toMetal(Program& program, OutputStream& out) {
Brian Osman7a20b5c2021-03-15 16:23:33 -0400965 TRACE_EVENT0("skia.shaders", "SkSL::Compiler::toMetal");
John Stilesa289ac22021-05-06 07:35:35 -0400966 AutoSource as(this, program.fSource.get());
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400967 MetalCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholascc305772017-10-13 16:17:45 -0400968 bool result = cg.generateCode();
Ethan Nicholascc305772017-10-13 16:17:45 -0400969 return result;
970}
971
Ethan Nicholas00543112018-07-31 09:44:36 -0400972bool Compiler::toMetal(Program& program, String* out) {
Timothy Liangb8eeb802018-07-23 16:46:16 -0400973 StringStream buffer;
974 bool result = this->toMetal(program, buffer);
975 if (result) {
976 *out = buffer.str();
977 }
978 return result;
979}
980
Ethan Nicholas2a479a52020-08-18 16:29:45 -0400981#endif // defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
Brian Osman2e29ab52019-09-20 12:19:11 -0400982
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700983Position Compiler::position(int offset) {
Ethan Nicholas3c729892020-12-07 12:47:17 -0500984 if (fSource && offset >= 0) {
985 int line = 1;
986 int column = 1;
987 for (int i = 0; i < offset; i++) {
988 if ((*fSource)[i] == '\n') {
989 ++line;
990 column = 1;
991 }
992 else {
993 ++column;
994 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700995 }
Ethan Nicholas3c729892020-12-07 12:47:17 -0500996 return Position(line, column);
997 } else {
998 return Position(-1, -1);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700999 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001000}
1001
1002void Compiler::error(int offset, String msg) {
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001003 if (strstr(msg.c_str(), POISON_TAG)) {
1004 // don't report errors on poison values
1005 return;
1006 }
ethannicholasb3058bd2016-07-01 08:22:01 -07001007 fErrorCount++;
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001008 Position pos = this->position(offset);
John Stiles8d3642e2021-01-22 09:50:04 -05001009 fErrorTextLength.push_back(fErrorText.length());
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -04001010 if (!msg.starts_with("error: ")) {
1011 fErrorText += "error: ";
1012 }
1013 fErrorText += (pos.fLine >= 1 ? to_string(pos.fLine) + ": " : "") + msg + "\n";
ethannicholasb3058bd2016-07-01 08:22:01 -07001014}
1015
John Stiles8d3642e2021-01-22 09:50:04 -05001016void Compiler::setErrorCount(int c) {
1017 if (c < fErrorCount) {
1018 fErrorText.resize(fErrorTextLength[c]);
1019 fErrorTextLength.resize(c);
1020 fErrorCount = c;
1021 }
1022}
1023
Ethan Nicholas95046142021-01-07 10:57:27 -05001024String Compiler::errorText(bool showCount) {
1025 if (showCount) {
1026 this->writeErrorCount();
1027 }
Ethan Nicholas00543112018-07-31 09:44:36 -04001028 fErrorCount = 0;
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001029 String result = fErrorText;
Ethan Nicholas95046142021-01-07 10:57:27 -05001030 fErrorText = "";
ethannicholasb3058bd2016-07-01 08:22:01 -07001031 return result;
1032}
1033
1034void Compiler::writeErrorCount() {
1035 if (fErrorCount) {
1036 fErrorText += to_string(fErrorCount) + " error";
1037 if (fErrorCount > 1) {
1038 fErrorText += "s";
1039 }
1040 fErrorText += "\n";
1041 }
1042}
1043
John Stilesa6841be2020-08-06 14:11:56 -04001044} // namespace SkSL