blob: bad81e21408de084d59beb934f6bc9562134a653 [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 Stilesb92641c2020-08-31 18:09:01 -040013#include "src/sksl/SkSLAnalysis.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "src/sksl/SkSLByteCodeGenerator.h"
15#include "src/sksl/SkSLCFGGenerator.h"
16#include "src/sksl/SkSLCPPCodeGenerator.h"
17#include "src/sksl/SkSLGLSLCodeGenerator.h"
18#include "src/sksl/SkSLHCodeGenerator.h"
19#include "src/sksl/SkSLIRGenerator.h"
20#include "src/sksl/SkSLMetalCodeGenerator.h"
21#include "src/sksl/SkSLPipelineStageCodeGenerator.h"
Ethan Nicholasc18bb512020-07-28 14:46:53 -040022#include "src/sksl/SkSLRehydrator.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "src/sksl/SkSLSPIRVCodeGenerator.h"
Brian Osmanc0243912020-02-19 15:35:26 -050024#include "src/sksl/SkSLSPIRVtoHLSL.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050025#include "src/sksl/ir/SkSLEnum.h"
26#include "src/sksl/ir/SkSLExpression.h"
27#include "src/sksl/ir/SkSLExpressionStatement.h"
28#include "src/sksl/ir/SkSLFunctionCall.h"
29#include "src/sksl/ir/SkSLIntLiteral.h"
30#include "src/sksl/ir/SkSLModifiersDeclaration.h"
31#include "src/sksl/ir/SkSLNop.h"
32#include "src/sksl/ir/SkSLSymbolTable.h"
33#include "src/sksl/ir/SkSLTernaryExpression.h"
34#include "src/sksl/ir/SkSLUnresolvedFunction.h"
35#include "src/sksl/ir/SkSLVarDeclarations.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070036
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -040037#include <fstream>
38
Ethan Nicholasa11035b2019-11-26 16:27:47 -050039#if !defined(SKSL_STANDALONE) & SK_SUPPORT_GPU
40#include "include/gpu/GrContextOptions.h"
41#include "src/gpu/GrShaderCaps.h"
42#endif
43
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -040044#ifdef SK_ENABLE_SPIRV_VALIDATION
45#include "spirv-tools/libspirv.hpp"
46#endif
47
Brian Osmandd496172020-08-08 08:17:18 -040048#if !SKSL_STANDALONE
Ethan Nicholasc18bb512020-07-28 14:46:53 -040049
50#include "src/sksl/generated/sksl_fp.dehydrated.sksl"
51#include "src/sksl/generated/sksl_frag.dehydrated.sksl"
52#include "src/sksl/generated/sksl_geom.dehydrated.sksl"
53#include "src/sksl/generated/sksl_gpu.dehydrated.sksl"
54#include "src/sksl/generated/sksl_interp.dehydrated.sksl"
55#include "src/sksl/generated/sksl_pipeline.dehydrated.sksl"
56#include "src/sksl/generated/sksl_vert.dehydrated.sksl"
57
58#else
59
Brian Osmandd496172020-08-08 08:17:18 -040060// GN generates or copies all of these files to the skslc executable directory
61static const char SKSL_GPU_INCLUDE[] = "sksl_gpu.sksl";
62static const char SKSL_INTERP_INCLUDE[] = "sksl_interp.sksl";
63static const char SKSL_VERT_INCLUDE[] = "sksl_vert.sksl";
64static const char SKSL_FRAG_INCLUDE[] = "sksl_frag.sksl";
65static const char SKSL_GEOM_INCLUDE[] = "sksl_geom.sksl";
66static const char SKSL_FP_INCLUDE[] = "sksl_fp.sksl";
67static const char SKSL_PIPELINE_INCLUDE[] = "sksl_pipeline.sksl";
Ethan Nicholasc18bb512020-07-28 14:46:53 -040068
69#endif
Ethan Nicholas0d997662019-04-08 09:46:01 -040070
ethannicholasb3058bd2016-07-01 08:22:01 -070071namespace SkSL {
72
Ethan Nicholasdb80f692019-11-22 14:06:12 -050073static void grab_intrinsics(std::vector<std::unique_ptr<ProgramElement>>* src,
John Stiles810c8cf2020-08-26 19:46:27 -040074 IRIntrinsicMap* target) {
Brian Osman08f986d2020-05-13 17:06:46 -040075 for (auto iter = src->begin(); iter != src->end(); ) {
76 std::unique_ptr<ProgramElement>& element = *iter;
Ethan Nicholase6592142020-09-08 10:22:09 -040077 switch (element->kind()) {
78 case ProgramElement::Kind::kFunction: {
John Stiles3dc0da62020-08-19 17:48:31 -040079 FunctionDefinition& f = element->as<FunctionDefinition>();
Brian Osman08f986d2020-05-13 17:06:46 -040080 SkASSERT(f.fDeclaration.fBuiltin);
Ethan Nicholasc18bb512020-07-28 14:46:53 -040081 String key = f.fDeclaration.description();
Brian Osman08f986d2020-05-13 17:06:46 -040082 SkASSERT(target->find(key) == target->end());
John Stiles810c8cf2020-08-26 19:46:27 -040083 (*target)[key] = IRIntrinsic{std::move(element), /*fAlreadyIncluded=*/false};
Brian Osman08f986d2020-05-13 17:06:46 -040084 iter = src->erase(iter);
Ethan Nicholasdb80f692019-11-22 14:06:12 -050085 break;
86 }
Ethan Nicholase6592142020-09-08 10:22:09 -040087 case ProgramElement::Kind::kEnum: {
John Stiles3dc0da62020-08-19 17:48:31 -040088 Enum& e = element->as<Enum>();
Ethan Nicholasdb80f692019-11-22 14:06:12 -050089 StringFragment name = e.fTypeName;
90 SkASSERT(target->find(name) == target->end());
John Stiles810c8cf2020-08-26 19:46:27 -040091 (*target)[name] = IRIntrinsic{std::move(element), /*fAlreadyIncluded=*/false};
Brian Osman08f986d2020-05-13 17:06:46 -040092 iter = src->erase(iter);
Ethan Nicholasdb80f692019-11-22 14:06:12 -050093 break;
94 }
95 default:
96 printf("unsupported include file element\n");
97 SkASSERT(false);
98 }
99 }
100}
101
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -0400102Compiler::Compiler(Flags flags)
John Stiles810c8cf2020-08-26 19:46:27 -0400103: fGPUIntrinsics(std::make_unique<IRIntrinsicMap>())
104, fInterpreterIntrinsics(std::make_unique<IRIntrinsicMap>())
105, fFlags(flags)
106, fContext(std::make_shared<Context>())
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -0400107, fErrorCount(0) {
Brian Osmaneac49832020-09-18 11:49:22 -0400108 fRootSymbolTable = std::make_shared<SymbolTable>(this);
John Stiles941fc712020-09-19 12:47:10 +0000109 fIRGenerator =
110 std::make_unique<IRGenerator>(fContext.get(), &fInliner, fRootSymbolTable, *this);
Brian Osmaneac49832020-09-18 11:49:22 -0400111 #define ADD_TYPE(t) fRootSymbolTable->addWithoutOwnership(fContext->f ## t ## _Type->fName, \
112 fContext->f ## t ## _Type.get())
ethannicholasb3058bd2016-07-01 08:22:01 -0700113 ADD_TYPE(Void);
114 ADD_TYPE(Float);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400115 ADD_TYPE(Float2);
116 ADD_TYPE(Float3);
117 ADD_TYPE(Float4);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400118 ADD_TYPE(Half);
119 ADD_TYPE(Half2);
120 ADD_TYPE(Half3);
121 ADD_TYPE(Half4);
ethannicholasb3058bd2016-07-01 08:22:01 -0700122 ADD_TYPE(Int);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400123 ADD_TYPE(Int2);
124 ADD_TYPE(Int3);
125 ADD_TYPE(Int4);
ethannicholasb3058bd2016-07-01 08:22:01 -0700126 ADD_TYPE(UInt);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400127 ADD_TYPE(UInt2);
128 ADD_TYPE(UInt3);
129 ADD_TYPE(UInt4);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400130 ADD_TYPE(Short);
131 ADD_TYPE(Short2);
132 ADD_TYPE(Short3);
133 ADD_TYPE(Short4);
134 ADD_TYPE(UShort);
135 ADD_TYPE(UShort2);
136 ADD_TYPE(UShort3);
137 ADD_TYPE(UShort4);
Ruiqi Maob609e6d2018-07-17 10:19:38 -0400138 ADD_TYPE(Byte);
139 ADD_TYPE(Byte2);
140 ADD_TYPE(Byte3);
141 ADD_TYPE(Byte4);
142 ADD_TYPE(UByte);
143 ADD_TYPE(UByte2);
144 ADD_TYPE(UByte3);
145 ADD_TYPE(UByte4);
ethannicholasb3058bd2016-07-01 08:22:01 -0700146 ADD_TYPE(Bool);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400147 ADD_TYPE(Bool2);
148 ADD_TYPE(Bool3);
149 ADD_TYPE(Bool4);
150 ADD_TYPE(Float2x2);
151 ADD_TYPE(Float2x3);
152 ADD_TYPE(Float2x4);
153 ADD_TYPE(Float3x2);
154 ADD_TYPE(Float3x3);
155 ADD_TYPE(Float3x4);
156 ADD_TYPE(Float4x2);
157 ADD_TYPE(Float4x3);
158 ADD_TYPE(Float4x4);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400159 ADD_TYPE(Half2x2);
160 ADD_TYPE(Half2x3);
161 ADD_TYPE(Half2x4);
162 ADD_TYPE(Half3x2);
163 ADD_TYPE(Half3x3);
164 ADD_TYPE(Half3x4);
165 ADD_TYPE(Half4x2);
166 ADD_TYPE(Half4x3);
167 ADD_TYPE(Half4x4);
ethannicholasb3058bd2016-07-01 08:22:01 -0700168 ADD_TYPE(GenType);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400169 ADD_TYPE(GenHType);
ethannicholasb3058bd2016-07-01 08:22:01 -0700170 ADD_TYPE(GenIType);
171 ADD_TYPE(GenUType);
172 ADD_TYPE(GenBType);
173 ADD_TYPE(Mat);
174 ADD_TYPE(Vec);
175 ADD_TYPE(GVec);
176 ADD_TYPE(GVec2);
177 ADD_TYPE(GVec3);
178 ADD_TYPE(GVec4);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400179 ADD_TYPE(HVec);
ethannicholasb3058bd2016-07-01 08:22:01 -0700180 ADD_TYPE(IVec);
181 ADD_TYPE(UVec);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400182 ADD_TYPE(SVec);
183 ADD_TYPE(USVec);
Ruiqi Maob609e6d2018-07-17 10:19:38 -0400184 ADD_TYPE(ByteVec);
185 ADD_TYPE(UByteVec);
ethannicholasb3058bd2016-07-01 08:22:01 -0700186 ADD_TYPE(BVec);
187
188 ADD_TYPE(Sampler1D);
189 ADD_TYPE(Sampler2D);
190 ADD_TYPE(Sampler3D);
ethannicholas5961bc92016-10-12 06:39:56 -0700191 ADD_TYPE(SamplerExternalOES);
ethannicholasb3058bd2016-07-01 08:22:01 -0700192 ADD_TYPE(SamplerCube);
193 ADD_TYPE(Sampler2DRect);
194 ADD_TYPE(Sampler1DArray);
195 ADD_TYPE(Sampler2DArray);
196 ADD_TYPE(SamplerCubeArray);
197 ADD_TYPE(SamplerBuffer);
198 ADD_TYPE(Sampler2DMS);
199 ADD_TYPE(Sampler2DMSArray);
200
Brian Salomonbf7b6202016-11-11 16:08:03 -0500201 ADD_TYPE(ISampler2D);
202
Brian Salomon2a51de82016-11-16 12:06:01 -0500203 ADD_TYPE(Image2D);
204 ADD_TYPE(IImage2D);
205
Greg Daniel64773e62016-11-22 09:44:03 -0500206 ADD_TYPE(SubpassInput);
207 ADD_TYPE(SubpassInputMS);
208
ethannicholasb3058bd2016-07-01 08:22:01 -0700209 ADD_TYPE(GSampler1D);
210 ADD_TYPE(GSampler2D);
211 ADD_TYPE(GSampler3D);
212 ADD_TYPE(GSamplerCube);
213 ADD_TYPE(GSampler2DRect);
214 ADD_TYPE(GSampler1DArray);
215 ADD_TYPE(GSampler2DArray);
216 ADD_TYPE(GSamplerCubeArray);
217 ADD_TYPE(GSamplerBuffer);
218 ADD_TYPE(GSampler2DMS);
219 ADD_TYPE(GSampler2DMSArray);
220
221 ADD_TYPE(Sampler1DShadow);
222 ADD_TYPE(Sampler2DShadow);
223 ADD_TYPE(SamplerCubeShadow);
224 ADD_TYPE(Sampler2DRectShadow);
225 ADD_TYPE(Sampler1DArrayShadow);
226 ADD_TYPE(Sampler2DArrayShadow);
227 ADD_TYPE(SamplerCubeArrayShadow);
228 ADD_TYPE(GSampler2DArrayShadow);
229 ADD_TYPE(GSamplerCubeArrayShadow);
Ethan Nicholasc9472af2017-10-10 16:30:21 -0400230 ADD_TYPE(FragmentProcessor);
Stephen Whiteff5d7a22019-07-26 17:42:06 -0400231 ADD_TYPE(Sampler);
232 ADD_TYPE(Texture2D);
ethannicholasb3058bd2016-07-01 08:22:01 -0700233
Brian Osman28590d52020-03-23 16:59:08 -0400234 StringFragment fpAliasName("shader");
Brian Osmaneac49832020-09-18 11:49:22 -0400235 fRootSymbolTable->addWithoutOwnership(fpAliasName, fContext->fFragmentProcessor_Type.get());
Brian Osman28590d52020-03-23 16:59:08 -0400236
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700237 StringFragment skCapsName("sk_Caps");
Brian Osmaneac49832020-09-18 11:49:22 -0400238 fRootSymbolTable->add(
John Stiles311dd9d2020-08-13 17:09:29 -0400239 skCapsName,
240 std::make_unique<Variable>(/*offset=*/-1, Modifiers(), skCapsName,
Ethan Nicholas30d30222020-09-11 12:27:26 -0400241 fContext->fSkCaps_Type.get(), Variable::kGlobal_Storage));
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500242
John Stiles810c8cf2020-08-26 19:46:27 -0400243 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholasdb80f692019-11-22 14:06:12 -0500244 std::vector<std::unique_ptr<ProgramElement>> gpuIntrinsics;
Brian Osmandd496172020-08-08 08:17:18 -0400245#if SKSL_STANDALONE
Brian Osmaneac49832020-09-18 11:49:22 -0400246 this->processIncludeFile(Program::kFragment_Kind, SKSL_GPU_INCLUDE, fRootSymbolTable,
247 &gpuIntrinsics, &fGpuSymbolTable);
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400248 this->processIncludeFile(Program::kVertex_Kind, SKSL_VERT_INCLUDE, fGpuSymbolTable,
249 &fVertexInclude, &fVertexSymbolTable);
250 this->processIncludeFile(Program::kFragment_Kind, SKSL_FRAG_INCLUDE, fGpuSymbolTable,
251 &fFragmentInclude, &fFragmentSymbolTable);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400252#else
253 {
Brian Osmaneac49832020-09-18 11:49:22 -0400254 Rehydrator rehydrator(fContext.get(), fRootSymbolTable, this, SKSL_INCLUDE_sksl_gpu,
255 SKSL_INCLUDE_sksl_gpu_LENGTH);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400256 fGpuSymbolTable = rehydrator.symbolTable();
257 gpuIntrinsics = rehydrator.elements();
258 }
259 {
260 Rehydrator rehydrator(fContext.get(), fGpuSymbolTable, this, SKSL_INCLUDE_sksl_vert,
Brian Osmaneac49832020-09-18 11:49:22 -0400261 SKSL_INCLUDE_sksl_vert_LENGTH);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400262 fVertexSymbolTable = rehydrator.symbolTable();
263 fVertexInclude = rehydrator.elements();
264 }
265 {
266 Rehydrator rehydrator(fContext.get(), fGpuSymbolTable, this, SKSL_INCLUDE_sksl_frag,
Brian Osmaneac49832020-09-18 11:49:22 -0400267 SKSL_INCLUDE_sksl_frag_LENGTH);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400268 fFragmentSymbolTable = rehydrator.symbolTable();
269 fFragmentInclude = rehydrator.elements();
270 }
271#endif
John Stiles810c8cf2020-08-26 19:46:27 -0400272 grab_intrinsics(&gpuIntrinsics, fGPUIntrinsics.get());
ethannicholasb3058bd2016-07-01 08:22:01 -0700273}
274
John Stiles656427a2020-08-27 15:26:26 -0400275Compiler::~Compiler() {}
ethannicholasb3058bd2016-07-01 08:22:01 -0700276
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400277void Compiler::loadGeometryIntrinsics() {
278 if (fGeometrySymbolTable) {
279 return;
280 }
Brian Osmandd496172020-08-08 08:17:18 -0400281 #if !SKSL_STANDALONE
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400282 {
283 Rehydrator rehydrator(fContext.get(), fGpuSymbolTable, this, SKSL_INCLUDE_sksl_geom,
284 SKSL_INCLUDE_sksl_geom_LENGTH);
285 fGeometrySymbolTable = rehydrator.symbolTable();
286 fGeometryInclude = rehydrator.elements();
287 }
288 #else
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400289 this->processIncludeFile(Program::kGeometry_Kind, SKSL_GEOM_INCLUDE, fGpuSymbolTable,
290 &fGeometryInclude, &fGeometrySymbolTable);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400291 #endif
292}
293
294void Compiler::loadPipelineIntrinsics() {
295 if (fPipelineSymbolTable) {
296 return;
297 }
Brian Osmandd496172020-08-08 08:17:18 -0400298 #if !SKSL_STANDALONE
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400299 {
300 Rehydrator rehydrator(fContext.get(), fGpuSymbolTable, this,
301 SKSL_INCLUDE_sksl_pipeline,
302 SKSL_INCLUDE_sksl_pipeline_LENGTH);
303 fPipelineSymbolTable = rehydrator.symbolTable();
304 fPipelineInclude = rehydrator.elements();
305 }
306 #else
307 this->processIncludeFile(Program::kPipelineStage_Kind, SKSL_PIPELINE_INCLUDE,
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400308 fGpuSymbolTable, &fPipelineInclude, &fPipelineSymbolTable);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400309 #endif
310}
311
312void Compiler::loadInterpreterIntrinsics() {
313 if (fInterpreterSymbolTable) {
314 return;
315 }
Brian Osmaneac49832020-09-18 11:49:22 -0400316 std::vector<std::unique_ptr<ProgramElement>> interpIntrinsics;
Brian Osmandd496172020-08-08 08:17:18 -0400317 #if !SKSL_STANDALONE
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400318 {
Brian Osmaneac49832020-09-18 11:49:22 -0400319 Rehydrator rehydrator(fContext.get(), fRootSymbolTable, this,
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400320 SKSL_INCLUDE_sksl_interp,
321 SKSL_INCLUDE_sksl_interp_LENGTH);
322 fInterpreterSymbolTable = rehydrator.symbolTable();
Brian Osmaneac49832020-09-18 11:49:22 -0400323 interpIntrinsics = rehydrator.elements();
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400324 }
325 #else
326 this->processIncludeFile(Program::kGeneric_Kind, SKSL_INTERP_INCLUDE,
Brian Osmaneac49832020-09-18 11:49:22 -0400327 fIRGenerator->fSymbolTable, &interpIntrinsics,
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400328 &fInterpreterSymbolTable);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400329 #endif
Brian Osmaneac49832020-09-18 11:49:22 -0400330 grab_intrinsics(&interpIntrinsics, fInterpreterIntrinsics.get());
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400331}
332
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400333void Compiler::processIncludeFile(Program::Kind kind, const char* path,
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400334 std::shared_ptr<SymbolTable> base,
335 std::vector<std::unique_ptr<ProgramElement>>* outElements,
336 std::shared_ptr<SymbolTable>* outSymbolTable) {
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400337 std::ifstream in(path);
338 std::string stdText{std::istreambuf_iterator<char>(in),
339 std::istreambuf_iterator<char>()};
340 if (in.rdstate()) {
341 printf("error reading %s\n", path);
342 abort();
343 }
344 if (!base) {
345 base = fIRGenerator->fSymbolTable;
346 }
347 SkASSERT(base);
348 const String* source = base->takeOwnershipOfString(std::make_unique<String>(stdText.c_str()));
349 fSource = source;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400350 std::shared_ptr<SymbolTable> old = fIRGenerator->fSymbolTable;
351 if (base) {
352 fIRGenerator->fSymbolTable = std::move(base);
353 }
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400354 Program::Settings settings;
John Stiles910845f2020-09-18 15:41:07 -0400355 settings.fInline = false;
Ethan Nicholasa11035b2019-11-26 16:27:47 -0500356#if !defined(SKSL_STANDALONE) & SK_SUPPORT_GPU
357 GrContextOptions opts;
358 GrShaderCaps caps(opts);
359 settings.fCaps = &caps;
360#endif
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400361 fIRGenerator->start(&settings, nullptr, true);
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -0400362 fIRGenerator->convertProgram(kind, source->c_str(), source->length(), outElements);
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400363 if (this->fErrorCount) {
364 printf("Unexpected errors: %s\n", this->fErrorText.c_str());
365 }
366 SkASSERT(!fErrorCount);
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400367 *outSymbolTable = fIRGenerator->fSymbolTable;
Ethan Nicholasa11035b2019-11-26 16:27:47 -0500368#ifdef SK_DEBUG
369 fSource = nullptr;
370#endif
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400371 fIRGenerator->fSymbolTable = std::move(old);
Ethan Nicholas8da1e652019-05-24 11:01:59 -0400372}
373
ethannicholas22f939e2016-10-13 13:25:34 -0700374// add the definition created by assigning to the lvalue to the definition set
Ethan Nicholas86a43402017-01-19 13:32:00 -0500375void Compiler::addDefinition(const Expression* lvalue, std::unique_ptr<Expression>* expr,
376 DefinitionMap* definitions) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400377 switch (lvalue->kind()) {
378 case Expression::Kind::kVariableReference: {
John Stilesa5a97b42020-08-18 11:19:07 -0400379 const Variable& var = lvalue->as<VariableReference>().fVariable;
ethannicholas22f939e2016-10-13 13:25:34 -0700380 if (var.fStorage == Variable::kLocal_Storage) {
381 (*definitions)[&var] = expr;
382 }
383 break;
384 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400385 case Expression::Kind::kSwizzle:
ethannicholas22f939e2016-10-13 13:25:34 -0700386 // We consider the variable written to as long as at least some of its components have
387 // been written to. This will lead to some false negatives (we won't catch it if you
388 // write to foo.x and then read foo.y), but being stricter could lead to false positives
Mike Klein6ad99092016-10-26 10:35:22 -0400389 // (we write to foo.x, and then pass foo to a function which happens to only read foo.x,
390 // but since we pass foo as a whole it is flagged as an error) unless we perform a much
ethannicholas22f939e2016-10-13 13:25:34 -0700391 // more complicated whole-program analysis. This is probably good enough.
John Stilesa5a97b42020-08-18 11:19:07 -0400392 this->addDefinition(lvalue->as<Swizzle>().fBase.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400393 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700394 definitions);
395 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400396 case Expression::Kind::kIndex:
ethannicholas22f939e2016-10-13 13:25:34 -0700397 // see comments in Swizzle
John Stilesa5a97b42020-08-18 11:19:07 -0400398 this->addDefinition(lvalue->as<IndexExpression>().fBase.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400399 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700400 definitions);
401 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400402 case Expression::Kind::kFieldAccess:
ethannicholas22f939e2016-10-13 13:25:34 -0700403 // see comments in Swizzle
John Stilesa5a97b42020-08-18 11:19:07 -0400404 this->addDefinition(lvalue->as<FieldAccess>().fBase.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400405 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700406 definitions);
407 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400408 case Expression::Kind::kTernary:
Ethan Nicholasa583b812018-01-18 13:32:11 -0500409 // To simplify analysis, we just pretend that we write to both sides of the ternary.
410 // This allows for false positives (meaning we fail to detect that a variable might not
411 // have been assigned), but is preferable to false negatives.
John Stilesa5a97b42020-08-18 11:19:07 -0400412 this->addDefinition(lvalue->as<TernaryExpression>().fIfTrue.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400413 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
Ethan Nicholasa583b812018-01-18 13:32:11 -0500414 definitions);
John Stilesa5a97b42020-08-18 11:19:07 -0400415 this->addDefinition(lvalue->as<TernaryExpression>().fIfFalse.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400416 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
Ethan Nicholasa583b812018-01-18 13:32:11 -0500417 definitions);
418 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400419 case Expression::Kind::kExternalValue:
Ethan Nicholas91164d12019-05-15 15:29:54 -0400420 break;
ethannicholas22f939e2016-10-13 13:25:34 -0700421 default:
422 // not an lvalue, can't happen
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400423 SkASSERT(false);
ethannicholas22f939e2016-10-13 13:25:34 -0700424 }
425}
426
427// add local variables defined by this node to the set
Mike Klein6ad99092016-10-26 10:35:22 -0400428void Compiler::addDefinitions(const BasicBlock::Node& node,
Ethan Nicholas86a43402017-01-19 13:32:00 -0500429 DefinitionMap* definitions) {
ethannicholas22f939e2016-10-13 13:25:34 -0700430 switch (node.fKind) {
431 case BasicBlock::Node::kExpression_Kind: {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400432 SkASSERT(node.expression());
John Stilesa5a97b42020-08-18 11:19:07 -0400433 Expression* expr = node.expression()->get();
Ethan Nicholase6592142020-09-08 10:22:09 -0400434 switch (expr->kind()) {
435 case Expression::Kind::kBinary: {
John Stilesa5a97b42020-08-18 11:19:07 -0400436 BinaryExpression* b = &expr->as<BinaryExpression>();
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000437 if (b->fOperator == Token::Kind::TK_EQ) {
438 this->addDefinition(b->fLeft.get(), &b->fRight, definitions);
439 } else if (Compiler::IsAssignment(b->fOperator)) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500440 this->addDefinition(
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000441 b->fLeft.get(),
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400442 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
443 definitions);
Ethan Nicholas86a43402017-01-19 13:32:00 -0500444
445 }
446 break;
ethannicholas22f939e2016-10-13 13:25:34 -0700447 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400448 case Expression::Kind::kFunctionCall: {
John Stilesa5a97b42020-08-18 11:19:07 -0400449 const FunctionCall& c = expr->as<FunctionCall>();
Ethan Nicholasc6a19f12018-03-29 16:46:56 -0400450 for (size_t i = 0; i < c.fFunction.fParameters.size(); ++i) {
451 if (c.fFunction.fParameters[i]->fModifiers.fFlags & Modifiers::kOut_Flag) {
452 this->addDefinition(
453 c.fArguments[i].get(),
454 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
455 definitions);
456 }
457 }
458 break;
459 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400460 case Expression::Kind::kPrefix: {
John Stilesa5a97b42020-08-18 11:19:07 -0400461 const PrefixExpression* p = &expr->as<PrefixExpression>();
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -0400462 if (p->fOperator == Token::Kind::TK_MINUSMINUS ||
463 p->fOperator == Token::Kind::TK_PLUSPLUS) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500464 this->addDefinition(
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400465 p->fOperand.get(),
466 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
467 definitions);
Ethan Nicholas86a43402017-01-19 13:32:00 -0500468 }
469 break;
470 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400471 case Expression::Kind::kPostfix: {
John Stilesa5a97b42020-08-18 11:19:07 -0400472 const PostfixExpression* p = &expr->as<PostfixExpression>();
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -0400473 if (p->fOperator == Token::Kind::TK_MINUSMINUS ||
474 p->fOperator == Token::Kind::TK_PLUSPLUS) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500475 this->addDefinition(
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400476 p->fOperand.get(),
477 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
478 definitions);
Ethan Nicholas86a43402017-01-19 13:32:00 -0500479 }
480 break;
481 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400482 case Expression::Kind::kVariableReference: {
John Stilesa5a97b42020-08-18 11:19:07 -0400483 const VariableReference* v = &expr->as<VariableReference>();
Ethan Nicholascb670962017-04-20 19:31:52 -0400484 if (v->fRefKind != VariableReference::kRead_RefKind) {
485 this->addDefinition(
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400486 v,
487 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
488 definitions);
Ethan Nicholascb670962017-04-20 19:31:52 -0400489 }
John Stiles30212b72020-06-11 17:55:07 -0400490 break;
Ethan Nicholascb670962017-04-20 19:31:52 -0400491 }
Ethan Nicholas86a43402017-01-19 13:32:00 -0500492 default:
493 break;
ethannicholas22f939e2016-10-13 13:25:34 -0700494 }
495 break;
496 }
497 case BasicBlock::Node::kStatement_Kind: {
John Stilesa5a97b42020-08-18 11:19:07 -0400498 Statement* stmt = node.statement()->get();
Ethan Nicholase6592142020-09-08 10:22:09 -0400499 if (stmt->kind() == Statement::Kind::kVarDeclaration) {
John Stilesa5a97b42020-08-18 11:19:07 -0400500 VarDeclaration& vd = stmt->as<VarDeclaration>();
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000501 if (vd.fValue) {
502 (*definitions)[vd.fVar] = &vd.fValue;
ethannicholas22f939e2016-10-13 13:25:34 -0700503 }
504 }
505 break;
506 }
507 }
508}
509
510void Compiler::scanCFG(CFG* cfg, BlockId blockId, std::set<BlockId>* workList) {
511 BasicBlock& block = cfg->fBlocks[blockId];
512
513 // compute definitions after this block
Ethan Nicholas86a43402017-01-19 13:32:00 -0500514 DefinitionMap after = block.fBefore;
ethannicholas22f939e2016-10-13 13:25:34 -0700515 for (const BasicBlock::Node& n : block.fNodes) {
516 this->addDefinitions(n, &after);
517 }
518
519 // propagate definitions to exits
520 for (BlockId exitId : block.fExits) {
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400521 if (exitId == blockId) {
522 continue;
523 }
ethannicholas22f939e2016-10-13 13:25:34 -0700524 BasicBlock& exit = cfg->fBlocks[exitId];
525 for (const auto& pair : after) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500526 std::unique_ptr<Expression>* e1 = pair.second;
527 auto found = exit.fBefore.find(pair.first);
528 if (found == exit.fBefore.end()) {
529 // exit has no definition for it, just copy it
530 workList->insert(exitId);
ethannicholas22f939e2016-10-13 13:25:34 -0700531 exit.fBefore[pair.first] = e1;
532 } else {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500533 // exit has a (possibly different) value already defined
534 std::unique_ptr<Expression>* e2 = exit.fBefore[pair.first];
ethannicholas22f939e2016-10-13 13:25:34 -0700535 if (e1 != e2) {
536 // definition has changed, merge and add exit block to worklist
537 workList->insert(exitId);
Ethan Nicholasaf197692017-02-27 13:26:45 -0500538 if (e1 && e2) {
539 exit.fBefore[pair.first] =
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400540 (std::unique_ptr<Expression>*) &fContext->fDefined_Expression;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500541 } else {
542 exit.fBefore[pair.first] = nullptr;
543 }
ethannicholas22f939e2016-10-13 13:25:34 -0700544 }
545 }
546 }
547 }
548}
549
550// returns a map which maps all local variables in the function to null, indicating that their value
551// is initially unknown
Ethan Nicholas86a43402017-01-19 13:32:00 -0500552static DefinitionMap compute_start_state(const CFG& cfg) {
553 DefinitionMap result;
Mike Klein6ad99092016-10-26 10:35:22 -0400554 for (const auto& block : cfg.fBlocks) {
555 for (const auto& node : block.fNodes) {
ethannicholas22f939e2016-10-13 13:25:34 -0700556 if (node.fKind == BasicBlock::Node::kStatement_Kind) {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400557 SkASSERT(node.statement());
Ethan Nicholascb670962017-04-20 19:31:52 -0400558 const Statement* s = node.statement()->get();
Ethan Nicholase6592142020-09-08 10:22:09 -0400559 if (s->kind() == Statement::Kind::kVarDeclarations) {
John Stilesa5a97b42020-08-18 11:19:07 -0400560 const VarDeclarationsStatement* vd = &s->as<VarDeclarationsStatement>();
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000561 for (const auto& decl : vd->fDeclaration->fVars) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400562 if (decl->kind() == Statement::Kind::kVarDeclaration) {
John Stilesa5a97b42020-08-18 11:19:07 -0400563 result[decl->as<VarDeclaration>().fVar] = nullptr;
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000564 }
Mike Klein6ad99092016-10-26 10:35:22 -0400565 }
ethannicholas22f939e2016-10-13 13:25:34 -0700566 }
567 }
568 }
569 }
570 return result;
571}
572
Ethan Nicholascb670962017-04-20 19:31:52 -0400573/**
574 * Returns true if assigning to this lvalue has no effect.
575 */
576static bool is_dead(const Expression& lvalue) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400577 switch (lvalue.kind()) {
578 case Expression::Kind::kVariableReference:
John Stilesa5a97b42020-08-18 11:19:07 -0400579 return lvalue.as<VariableReference>().fVariable.dead();
Ethan Nicholase6592142020-09-08 10:22:09 -0400580 case Expression::Kind::kSwizzle:
John Stilesa5a97b42020-08-18 11:19:07 -0400581 return is_dead(*lvalue.as<Swizzle>().fBase);
Ethan Nicholase6592142020-09-08 10:22:09 -0400582 case Expression::Kind::kFieldAccess:
John Stilesa5a97b42020-08-18 11:19:07 -0400583 return is_dead(*lvalue.as<FieldAccess>().fBase);
Ethan Nicholase6592142020-09-08 10:22:09 -0400584 case Expression::Kind::kIndex: {
John Stilesa5a97b42020-08-18 11:19:07 -0400585 const IndexExpression& idx = lvalue.as<IndexExpression>();
Ethan Nicholas2a099da2020-01-02 14:40:54 -0500586 return is_dead(*idx.fBase) &&
587 !idx.fIndex->hasProperty(Expression::Property::kSideEffects);
Ethan Nicholascb670962017-04-20 19:31:52 -0400588 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400589 case Expression::Kind::kTernary: {
John Stilesa5a97b42020-08-18 11:19:07 -0400590 const TernaryExpression& t = lvalue.as<TernaryExpression>();
Ethan Nicholasa583b812018-01-18 13:32:11 -0500591 return !t.fTest->hasSideEffects() && is_dead(*t.fIfTrue) && is_dead(*t.fIfFalse);
592 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400593 case Expression::Kind::kExternalValue:
Ethan Nicholas91164d12019-05-15 15:29:54 -0400594 return false;
Ethan Nicholascb670962017-04-20 19:31:52 -0400595 default:
Ethan Nicholas2a099da2020-01-02 14:40:54 -0500596#ifdef SK_DEBUG
Ethan Nicholascb670962017-04-20 19:31:52 -0400597 ABORT("invalid lvalue: %s\n", lvalue.description().c_str());
Ethan Nicholas2a099da2020-01-02 14:40:54 -0500598#endif
599 return false;
Ethan Nicholascb670962017-04-20 19:31:52 -0400600 }
601}
ethannicholas22f939e2016-10-13 13:25:34 -0700602
Ethan Nicholascb670962017-04-20 19:31:52 -0400603/**
604 * Returns true if this is an assignment which can be collapsed down to just the right hand side due
605 * to a dead target and lack of side effects on the left hand side.
606 */
607static bool dead_assignment(const BinaryExpression& b) {
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000608 if (!Compiler::IsAssignment(b.fOperator)) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400609 return false;
610 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000611 return is_dead(*b.fLeft);
Ethan Nicholascb670962017-04-20 19:31:52 -0400612}
613
614void Compiler::computeDataFlow(CFG* cfg) {
615 cfg->fBlocks[cfg->fStart].fBefore = compute_start_state(*cfg);
ethannicholas22f939e2016-10-13 13:25:34 -0700616 std::set<BlockId> workList;
Ethan Nicholascb670962017-04-20 19:31:52 -0400617 for (BlockId i = 0; i < cfg->fBlocks.size(); i++) {
ethannicholas22f939e2016-10-13 13:25:34 -0700618 workList.insert(i);
619 }
620 while (workList.size()) {
621 BlockId next = *workList.begin();
622 workList.erase(workList.begin());
Ethan Nicholascb670962017-04-20 19:31:52 -0400623 this->scanCFG(cfg, next, &workList);
ethannicholas22f939e2016-10-13 13:25:34 -0700624 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400625}
626
627/**
628 * Attempts to replace the expression pointed to by iter with a new one (in both the CFG and the
629 * IR). If the expression can be cleanly removed, returns true and updates the iterator to point to
630 * the newly-inserted element. Otherwise updates only the IR and returns false (and the CFG will
631 * need to be regenerated).
632 */
John Stilesafbf8992020-08-18 10:08:21 -0400633static bool try_replace_expression(BasicBlock* b,
634 std::vector<BasicBlock::Node>::iterator* iter,
635 std::unique_ptr<Expression>* newExpression) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400636 std::unique_ptr<Expression>* target = (*iter)->expression();
637 if (!b->tryRemoveExpression(iter)) {
638 *target = std::move(*newExpression);
639 return false;
640 }
641 *target = std::move(*newExpression);
642 return b->tryInsertExpression(iter, target);
643}
644
645/**
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400646 * Returns true if the expression is a constant numeric literal with the specified value, or a
647 * constant vector with all elements equal to the specified value.
Ethan Nicholascb670962017-04-20 19:31:52 -0400648 */
John Stiles9d944232020-08-19 09:56:49 -0400649template <typename T = double>
650static bool is_constant(const Expression& expr, T value) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400651 switch (expr.kind()) {
652 case Expression::Kind::kIntLiteral:
John Stiles81365af2020-08-18 09:24:00 -0400653 return expr.as<IntLiteral>().fValue == value;
John Stiles9d944232020-08-19 09:56:49 -0400654
Ethan Nicholase6592142020-09-08 10:22:09 -0400655 case Expression::Kind::kFloatLiteral:
John Stiles81365af2020-08-18 09:24:00 -0400656 return expr.as<FloatLiteral>().fValue == value;
John Stiles9d944232020-08-19 09:56:49 -0400657
Ethan Nicholase6592142020-09-08 10:22:09 -0400658 case Expression::Kind::kConstructor: {
John Stiles9d944232020-08-19 09:56:49 -0400659 const Constructor& constructor = expr.as<Constructor>();
660 if (constructor.isCompileTimeConstant()) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400661 const Type& constructorType = constructor.type();
662 bool isFloat = constructorType.columns() > 1
663 ? constructorType.componentType().isFloat()
664 : constructorType.isFloat();
665 switch (constructorType.typeKind()) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400666 case Type::TypeKind::kVector:
Ethan Nicholas30d30222020-09-11 12:27:26 -0400667 for (int i = 0; i < constructorType.columns(); ++i) {
John Stiles9d944232020-08-19 09:56:49 -0400668 if (isFloat) {
669 if (constructor.getFVecComponent(i) != value) {
670 return false;
671 }
672 } else {
673 if (constructor.getIVecComponent(i) != value) {
674 return false;
675 }
676 }
Ethan Nicholasd188c182019-06-10 15:55:38 -0400677 }
John Stiles9d944232020-08-19 09:56:49 -0400678 return true;
679
Ethan Nicholase6592142020-09-08 10:22:09 -0400680 case Type::TypeKind::kScalar:
John Stiles9d944232020-08-19 09:56:49 -0400681 SkASSERT(constructor.fArguments.size() == 1);
682 return is_constant<T>(*constructor.fArguments[0], value);
683
684 default:
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400685 return false;
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400686 }
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400687 }
688 return false;
689 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400690 default:
691 return false;
692 }
693}
694
695/**
696 * Collapses the binary expression pointed to by iter down to just the right side (in both the IR
697 * and CFG structures).
698 */
John Stilesafbf8992020-08-18 10:08:21 -0400699static void delete_left(BasicBlock* b,
700 std::vector<BasicBlock::Node>::iterator* iter,
701 bool* outUpdated,
702 bool* outNeedsRescan) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400703 *outUpdated = true;
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400704 std::unique_ptr<Expression>* target = (*iter)->expression();
John Stilesa5a97b42020-08-18 11:19:07 -0400705 BinaryExpression& bin = (*target)->as<BinaryExpression>();
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000706 SkASSERT(!bin.fLeft->hasSideEffects());
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400707 bool result;
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000708 if (bin.fOperator == Token::Kind::TK_EQ) {
709 result = b->tryRemoveLValueBefore(iter, bin.fLeft.get());
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400710 } else {
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000711 result = b->tryRemoveExpressionBefore(iter, bin.fLeft.get());
Ethan Nicholascb670962017-04-20 19:31:52 -0400712 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000713 *target = std::move(bin.fRight);
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400714 if (!result) {
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400715 *outNeedsRescan = true;
716 return;
717 }
718 if (*iter == b->fNodes.begin()) {
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400719 *outNeedsRescan = true;
720 return;
721 }
722 --(*iter);
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400723 if ((*iter)->fKind != BasicBlock::Node::kExpression_Kind ||
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000724 (*iter)->expression() != &bin.fRight) {
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400725 *outNeedsRescan = true;
726 return;
727 }
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400728 *iter = b->fNodes.erase(*iter);
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400729 SkASSERT((*iter)->expression() == target);
Ethan Nicholascb670962017-04-20 19:31:52 -0400730}
731
732/**
733 * Collapses the binary expression pointed to by iter down to just the left side (in both the IR and
734 * CFG structures).
735 */
John Stilesafbf8992020-08-18 10:08:21 -0400736static void delete_right(BasicBlock* b,
737 std::vector<BasicBlock::Node>::iterator* iter,
738 bool* outUpdated,
739 bool* outNeedsRescan) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400740 *outUpdated = true;
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400741 std::unique_ptr<Expression>* target = (*iter)->expression();
John Stilesa5a97b42020-08-18 11:19:07 -0400742 BinaryExpression& bin = (*target)->as<BinaryExpression>();
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000743 SkASSERT(!bin.fRight->hasSideEffects());
744 if (!b->tryRemoveExpressionBefore(iter, bin.fRight.get())) {
745 *target = std::move(bin.fLeft);
Ethan Nicholascb670962017-04-20 19:31:52 -0400746 *outNeedsRescan = true;
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400747 return;
Ethan Nicholascb670962017-04-20 19:31:52 -0400748 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000749 *target = std::move(bin.fLeft);
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400750 if (*iter == b->fNodes.begin()) {
751 *outNeedsRescan = true;
752 return;
753 }
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400754 --(*iter);
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400755 if (((*iter)->fKind != BasicBlock::Node::kExpression_Kind ||
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000756 (*iter)->expression() != &bin.fLeft)) {
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400757 *outNeedsRescan = true;
758 return;
759 }
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400760 *iter = b->fNodes.erase(*iter);
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400761 SkASSERT((*iter)->expression() == target);
Ethan Nicholascb670962017-04-20 19:31:52 -0400762}
763
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400764/**
765 * Constructs the specified type using a single argument.
766 */
Ethan Nicholas30d30222020-09-11 12:27:26 -0400767static std::unique_ptr<Expression> construct(const Type* type, std::unique_ptr<Expression> v) {
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400768 std::vector<std::unique_ptr<Expression>> args;
769 args.push_back(std::move(v));
Ethan Nicholase6592142020-09-08 10:22:09 -0400770 std::unique_ptr<Expression> result = std::make_unique<Constructor>(-1, type, std::move(args));
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400771 return result;
772}
773
774/**
775 * Used in the implementations of vectorize_left and vectorize_right. Given a vector type and an
776 * expression x, deletes the expression pointed to by iter and replaces it with <type>(x).
777 */
778static void vectorize(BasicBlock* b,
779 std::vector<BasicBlock::Node>::iterator* iter,
780 const Type& type,
781 std::unique_ptr<Expression>* otherExpression,
782 bool* outUpdated,
783 bool* outNeedsRescan) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400784 SkASSERT((*(*iter)->expression())->kind() == Expression::Kind::kBinary);
785 SkASSERT(type.typeKind() == Type::TypeKind::kVector);
Ethan Nicholas30d30222020-09-11 12:27:26 -0400786 SkASSERT((*otherExpression)->type().typeKind() == Type::TypeKind::kScalar);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400787 *outUpdated = true;
788 std::unique_ptr<Expression>* target = (*iter)->expression();
789 if (!b->tryRemoveExpression(iter)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400790 *target = construct(&type, std::move(*otherExpression));
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400791 *outNeedsRescan = true;
792 } else {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400793 *target = construct(&type, std::move(*otherExpression));
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400794 if (!b->tryInsertExpression(iter, target)) {
795 *outNeedsRescan = true;
796 }
797 }
798}
799
800/**
801 * Given a binary expression of the form x <op> vec<n>(y), deletes the right side and vectorizes the
802 * left to yield vec<n>(x).
803 */
804static void vectorize_left(BasicBlock* b,
805 std::vector<BasicBlock::Node>::iterator* iter,
806 bool* outUpdated,
807 bool* outNeedsRescan) {
John Stilesa5a97b42020-08-18 11:19:07 -0400808 BinaryExpression& bin = (*(*iter)->expression())->as<BinaryExpression>();
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000809 vectorize(b, iter, bin.fRight->type(), &bin.fLeft, outUpdated, outNeedsRescan);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400810}
811
812/**
813 * Given a binary expression of the form vec<n>(x) <op> y, deletes the left side and vectorizes the
814 * right to yield vec<n>(y).
815 */
816static void vectorize_right(BasicBlock* b,
817 std::vector<BasicBlock::Node>::iterator* iter,
818 bool* outUpdated,
819 bool* outNeedsRescan) {
John Stilesa5a97b42020-08-18 11:19:07 -0400820 BinaryExpression& bin = (*(*iter)->expression())->as<BinaryExpression>();
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000821 vectorize(b, iter, bin.fLeft->type(), &bin.fRight, outUpdated, outNeedsRescan);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400822}
823
824// Mark that an expression which we were writing to is no longer being written to
John Stilesa5a97b42020-08-18 11:19:07 -0400825static void clear_write(Expression& expr) {
Ethan Nicholase6592142020-09-08 10:22:09 -0400826 switch (expr.kind()) {
827 case Expression::Kind::kVariableReference: {
John Stilesa5a97b42020-08-18 11:19:07 -0400828 expr.as<VariableReference>().setRefKind(VariableReference::kRead_RefKind);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400829 break;
830 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400831 case Expression::Kind::kFieldAccess:
John Stilesa5a97b42020-08-18 11:19:07 -0400832 clear_write(*expr.as<FieldAccess>().fBase);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400833 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400834 case Expression::Kind::kSwizzle:
John Stilesa5a97b42020-08-18 11:19:07 -0400835 clear_write(*expr.as<Swizzle>().fBase);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400836 break;
Ethan Nicholase6592142020-09-08 10:22:09 -0400837 case Expression::Kind::kIndex:
John Stilesa5a97b42020-08-18 11:19:07 -0400838 clear_write(*expr.as<IndexExpression>().fBase);
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400839 break;
840 default:
841 ABORT("shouldn't be writing to this kind of expression\n");
842 break;
843 }
844}
845
Ethan Nicholascb670962017-04-20 19:31:52 -0400846void Compiler::simplifyExpression(DefinitionMap& definitions,
847 BasicBlock& b,
848 std::vector<BasicBlock::Node>::iterator* iter,
849 std::unordered_set<const Variable*>* undefinedVariables,
850 bool* outUpdated,
851 bool* outNeedsRescan) {
852 Expression* expr = (*iter)->expression()->get();
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400853 SkASSERT(expr);
Ethan Nicholascb670962017-04-20 19:31:52 -0400854 if ((*iter)->fConstantPropagation) {
855 std::unique_ptr<Expression> optimized = expr->constantPropagate(*fIRGenerator, definitions);
856 if (optimized) {
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400857 *outUpdated = true;
Ethan Nicholas30d30222020-09-11 12:27:26 -0400858 optimized = fIRGenerator->coerce(std::move(optimized), expr->type());
Ethan Nicholas1d881522020-09-11 09:32:54 -0400859 SkASSERT(optimized);
Ethan Nicholascb670962017-04-20 19:31:52 -0400860 if (!try_replace_expression(&b, iter, &optimized)) {
861 *outNeedsRescan = true;
Ethan Nicholas4b330df2017-05-17 10:52:55 -0400862 return;
Ethan Nicholascb670962017-04-20 19:31:52 -0400863 }
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400864 SkASSERT((*iter)->fKind == BasicBlock::Node::kExpression_Kind);
Ethan Nicholascb670962017-04-20 19:31:52 -0400865 expr = (*iter)->expression()->get();
Ethan Nicholascb670962017-04-20 19:31:52 -0400866 }
867 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400868 switch (expr->kind()) {
869 case Expression::Kind::kVariableReference: {
John Stilesa5a97b42020-08-18 11:19:07 -0400870 const VariableReference& ref = expr->as<VariableReference>();
Ethan Nicholas8f7e28f2018-03-26 14:24:27 -0400871 const Variable& var = ref.fVariable;
872 if (ref.refKind() != VariableReference::kWrite_RefKind &&
873 ref.refKind() != VariableReference::kPointer_RefKind &&
874 var.fStorage == Variable::kLocal_Storage && !definitions[&var] &&
Ethan Nicholascb670962017-04-20 19:31:52 -0400875 (*undefinedVariables).find(&var) == (*undefinedVariables).end()) {
876 (*undefinedVariables).insert(&var);
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000877 this->error(expr->fOffset,
878 "'" + var.fName + "' has not been assigned");
Ethan Nicholascb670962017-04-20 19:31:52 -0400879 }
880 break;
881 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400882 case Expression::Kind::kTernary: {
John Stiles403a3632020-08-20 12:11:48 -0400883 TernaryExpression* t = &expr->as<TernaryExpression>();
Ethan Nicholase6592142020-09-08 10:22:09 -0400884 if (t->fTest->kind() == Expression::Kind::kBoolLiteral) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400885 // ternary has a constant test, replace it with either the true or
886 // false branch
John Stiles403a3632020-08-20 12:11:48 -0400887 if (t->fTest->as<BoolLiteral>().fValue) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400888 (*iter)->setExpression(std::move(t->fIfTrue));
889 } else {
890 (*iter)->setExpression(std::move(t->fIfFalse));
891 }
892 *outUpdated = true;
893 *outNeedsRescan = true;
894 }
895 break;
896 }
Ethan Nicholase6592142020-09-08 10:22:09 -0400897 case Expression::Kind::kBinary: {
John Stiles403a3632020-08-20 12:11:48 -0400898 BinaryExpression* bin = &expr->as<BinaryExpression>();
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400899 if (dead_assignment(*bin)) {
900 delete_left(&b, iter, outUpdated, outNeedsRescan);
901 break;
902 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000903 const Type& leftType = bin->fLeft->type();
904 const Type& rightType = bin->fRight->type();
Ethan Nicholasc2371a42017-05-05 10:04:06 -0400905 // collapse useless expressions like x * 1 or x + 0
Ethan Nicholas30d30222020-09-11 12:27:26 -0400906 if (((leftType.typeKind() != Type::TypeKind::kScalar) &&
907 (leftType.typeKind() != Type::TypeKind::kVector)) ||
908 ((rightType.typeKind() != Type::TypeKind::kScalar) &&
909 (rightType.typeKind() != Type::TypeKind::kVector))) {
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400910 break;
911 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000912 switch (bin->fOperator) {
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -0400913 case Token::Kind::TK_STAR:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000914 if (is_constant(*bin->fLeft, 1)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400915 if (leftType.typeKind() == Type::TypeKind::kVector &&
916 rightType.typeKind() == Type::TypeKind::kScalar) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400917 // float4(1) * x -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400918 vectorize_right(&b, iter, outUpdated, outNeedsRescan);
919 } else {
920 // 1 * x -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400921 // 1 * float4(x) -> float4(x)
922 // float4(1) * float4(x) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400923 delete_left(&b, iter, outUpdated, outNeedsRescan);
924 }
925 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000926 else if (is_constant(*bin->fLeft, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400927 if (leftType.typeKind() == Type::TypeKind::kScalar &&
928 rightType.typeKind() == Type::TypeKind::kVector &&
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000929 !bin->fRight->hasSideEffects()) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400930 // 0 * float4(x) -> float4(0)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400931 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
932 } else {
933 // 0 * x -> 0
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400934 // float4(0) * x -> float4(0)
935 // float4(0) * float4(x) -> float4(0)
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000936 if (!bin->fRight->hasSideEffects()) {
Ethan Nicholas51493ee2017-12-11 12:34:33 -0500937 delete_right(&b, iter, outUpdated, outNeedsRescan);
938 }
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400939 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400940 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000941 else if (is_constant(*bin->fRight, 1)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400942 if (leftType.typeKind() == Type::TypeKind::kScalar &&
943 rightType.typeKind() == Type::TypeKind::kVector) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400944 // x * float4(1) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400945 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
946 } else {
947 // x * 1 -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400948 // float4(x) * 1 -> float4(x)
949 // float4(x) * float4(1) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400950 delete_right(&b, iter, outUpdated, outNeedsRescan);
951 }
952 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000953 else if (is_constant(*bin->fRight, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400954 if (leftType.typeKind() == Type::TypeKind::kVector &&
955 rightType.typeKind() == Type::TypeKind::kScalar &&
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000956 !bin->fLeft->hasSideEffects()) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400957 // float4(x) * 0 -> float4(0)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400958 vectorize_right(&b, iter, outUpdated, outNeedsRescan);
959 } else {
960 // x * 0 -> 0
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400961 // x * float4(0) -> float4(0)
962 // float4(x) * float4(0) -> float4(0)
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000963 if (!bin->fLeft->hasSideEffects()) {
Ethan Nicholas51493ee2017-12-11 12:34:33 -0500964 delete_left(&b, iter, outUpdated, outNeedsRescan);
965 }
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400966 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400967 }
968 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -0400969 case Token::Kind::TK_PLUS:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000970 if (is_constant(*bin->fLeft, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400971 if (leftType.typeKind() == Type::TypeKind::kVector &&
972 rightType.typeKind() == Type::TypeKind::kScalar) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400973 // float4(0) + x -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400974 vectorize_right(&b, iter, outUpdated, outNeedsRescan);
975 } else {
976 // 0 + x -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400977 // 0 + float4(x) -> float4(x)
978 // float4(0) + float4(x) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400979 delete_left(&b, iter, outUpdated, outNeedsRescan);
980 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000981 } else if (is_constant(*bin->fRight, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400982 if (leftType.typeKind() == Type::TypeKind::kScalar &&
983 rightType.typeKind() == Type::TypeKind::kVector) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400984 // x + float4(0) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400985 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
986 } else {
987 // x + 0 -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400988 // float4(x) + 0 -> float4(x)
989 // float4(x) + float4(0) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400990 delete_right(&b, iter, outUpdated, outNeedsRescan);
991 }
Ethan Nicholas56e42712017-04-21 10:23:37 -0400992 }
993 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -0400994 case Token::Kind::TK_MINUS:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +0000995 if (is_constant(*bin->fRight, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -0400996 if (leftType.typeKind() == Type::TypeKind::kScalar &&
997 rightType.typeKind() == Type::TypeKind::kVector) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400998 // x - float4(0) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -0400999 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
1000 } else {
1001 // x - 0 -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -04001002 // float4(x) - 0 -> float4(x)
1003 // float4(x) - float4(0) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001004 delete_right(&b, iter, outUpdated, outNeedsRescan);
1005 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001006 }
1007 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001008 case Token::Kind::TK_SLASH:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001009 if (is_constant(*bin->fRight, 1)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -04001010 if (leftType.typeKind() == Type::TypeKind::kScalar &&
1011 rightType.typeKind() == Type::TypeKind::kVector) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -04001012 // x / float4(1) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001013 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
1014 } else {
1015 // x / 1 -> x
Ethan Nicholas5af9ea32017-07-28 15:19:46 -04001016 // float4(x) / 1 -> float4(x)
1017 // float4(x) / float4(1) -> float4(x)
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001018 delete_right(&b, iter, outUpdated, outNeedsRescan);
1019 }
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001020 } else if (is_constant(*bin->fLeft, 0)) {
Ethan Nicholas30d30222020-09-11 12:27:26 -04001021 if (leftType.typeKind() == Type::TypeKind::kScalar &&
1022 rightType.typeKind() == Type::TypeKind::kVector &&
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001023 !bin->fRight->hasSideEffects()) {
Ethan Nicholas5af9ea32017-07-28 15:19:46 -04001024 // 0 / float4(x) -> float4(0)
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001025 vectorize_left(&b, iter, outUpdated, outNeedsRescan);
1026 } else {
1027 // 0 / x -> 0
Ethan Nicholas5af9ea32017-07-28 15:19:46 -04001028 // float4(0) / x -> float4(0)
1029 // float4(0) / float4(x) -> float4(0)
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001030 if (!bin->fRight->hasSideEffects()) {
Ethan Nicholas51493ee2017-12-11 12:34:33 -05001031 delete_right(&b, iter, outUpdated, outNeedsRescan);
1032 }
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001033 }
1034 }
1035 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001036 case Token::Kind::TK_PLUSEQ:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001037 if (is_constant(*bin->fRight, 0)) {
1038 clear_write(*bin->fLeft);
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001039 delete_right(&b, iter, outUpdated, outNeedsRescan);
1040 }
1041 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001042 case Token::Kind::TK_MINUSEQ:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001043 if (is_constant(*bin->fRight, 0)) {
1044 clear_write(*bin->fLeft);
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001045 delete_right(&b, iter, outUpdated, outNeedsRescan);
1046 }
1047 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001048 case Token::Kind::TK_STAREQ:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001049 if (is_constant(*bin->fRight, 1)) {
1050 clear_write(*bin->fLeft);
Ethan Nicholasfe53e582017-04-27 16:24:51 -04001051 delete_right(&b, iter, outUpdated, outNeedsRescan);
1052 }
1053 break;
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001054 case Token::Kind::TK_SLASHEQ:
Ethan Nicholasbf66ffb2020-09-16 22:05:10 +00001055 if (is_constant(*bin->fRight, 1)) {
1056 clear_write(*bin->fLeft);
Ethan Nicholascb670962017-04-20 19:31:52 -04001057 delete_right(&b, iter, outUpdated, outNeedsRescan);
1058 }
1059 break;
1060 default:
1061 break;
1062 }
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001063 break;
1064 }
Ethan Nicholase6592142020-09-08 10:22:09 -04001065 case Expression::Kind::kSwizzle: {
John Stiles403a3632020-08-20 12:11:48 -04001066 Swizzle& s = expr->as<Swizzle>();
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001067 // detect identity swizzles like foo.rgba
Ethan Nicholas30d30222020-09-11 12:27:26 -04001068 if ((int) s.fComponents.size() == s.fBase->type().columns()) {
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001069 bool identity = true;
1070 for (int i = 0; i < (int) s.fComponents.size(); ++i) {
1071 if (s.fComponents[i] != i) {
1072 identity = false;
1073 break;
1074 }
1075 }
1076 if (identity) {
1077 *outUpdated = true;
1078 if (!try_replace_expression(&b, iter, &s.fBase)) {
1079 *outNeedsRescan = true;
1080 return;
1081 }
1082 SkASSERT((*iter)->fKind == BasicBlock::Node::kExpression_Kind);
1083 break;
1084 }
1085 }
1086 // detect swizzles of swizzles, e.g. replace foo.argb.r000 with foo.a000
Ethan Nicholase6592142020-09-08 10:22:09 -04001087 if (s.fBase->kind() == Expression::Kind::kSwizzle) {
John Stilesa5a97b42020-08-18 11:19:07 -04001088 Swizzle& base = s.fBase->as<Swizzle>();
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001089 std::vector<int> final;
1090 for (int c : s.fComponents) {
Brian Osman25647672020-09-15 15:16:56 -04001091 final.push_back(base.fComponents[c]);
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001092 }
1093 *outUpdated = true;
1094 std::unique_ptr<Expression> replacement(new Swizzle(*fContext, base.fBase->clone(),
1095 std::move(final)));
1096 if (!try_replace_expression(&b, iter, &replacement)) {
1097 *outNeedsRescan = true;
1098 return;
1099 }
1100 SkASSERT((*iter)->fKind == BasicBlock::Node::kExpression_Kind);
Ethan Nicholas409f6f02019-09-17 12:34:39 -04001101 }
John Stiles30212b72020-06-11 17:55:07 -04001102 break;
Ethan Nicholascb670962017-04-20 19:31:52 -04001103 }
1104 default:
1105 break;
1106 }
1107}
1108
John Stiles92219b42020-06-15 12:32:24 -04001109// Returns true if this statement could potentially execute a break at the current level. We ignore
1110// nested loops and switches, since any breaks inside of them will merely break the loop / switch.
John Stilesb92641c2020-08-31 18:09:01 -04001111static bool contains_conditional_break(Statement& stmt) {
1112 class ContainsConditionalBreak : public ProgramVisitor {
1113 public:
1114 bool visitStatement(const Statement& stmt) override {
Ethan Nicholase6592142020-09-08 10:22:09 -04001115 switch (stmt.kind()) {
1116 case Statement::Kind::kBlock:
John Stilesb92641c2020-08-31 18:09:01 -04001117 return this->INHERITED::visitStatement(stmt);
1118
Ethan Nicholase6592142020-09-08 10:22:09 -04001119 case Statement::Kind::kBreak:
John Stilesb92641c2020-08-31 18:09:01 -04001120 return fInConditional > 0;
1121
Ethan Nicholase6592142020-09-08 10:22:09 -04001122 case Statement::Kind::kIf: {
John Stilesb92641c2020-08-31 18:09:01 -04001123 ++fInConditional;
1124 bool result = this->INHERITED::visitStatement(stmt);
1125 --fInConditional;
1126 return result;
1127 }
1128
1129 default:
1130 return false;
1131 }
1132 }
1133
1134 int fInConditional = 0;
1135 using INHERITED = ProgramVisitor;
1136 };
1137
1138 return ContainsConditionalBreak{}.visitStatement(stmt);
John Stiles92219b42020-06-15 12:32:24 -04001139}
1140
Ethan Nicholas5005a222018-08-24 13:06:27 -04001141// returns true if this statement definitely executes a break at the current level (we ignore
1142// nested loops and switches, since any breaks inside of them will merely break the loop / switch)
John Stilesb92641c2020-08-31 18:09:01 -04001143static bool contains_unconditional_break(Statement& stmt) {
1144 class ContainsUnconditionalBreak : public ProgramVisitor {
1145 public:
1146 bool visitStatement(const Statement& stmt) override {
Ethan Nicholase6592142020-09-08 10:22:09 -04001147 switch (stmt.kind()) {
1148 case Statement::Kind::kBlock:
John Stilesb92641c2020-08-31 18:09:01 -04001149 return this->INHERITED::visitStatement(stmt);
1150
Ethan Nicholase6592142020-09-08 10:22:09 -04001151 case Statement::Kind::kBreak:
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001152 return true;
John Stilesb92641c2020-08-31 18:09:01 -04001153
1154 default:
1155 return false;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001156 }
John Stilesb92641c2020-08-31 18:09:01 -04001157 }
John Stiles92219b42020-06-15 12:32:24 -04001158
John Stilesb92641c2020-08-31 18:09:01 -04001159 using INHERITED = ProgramVisitor;
1160 };
John Stiles92219b42020-06-15 12:32:24 -04001161
John Stilesb92641c2020-08-31 18:09:01 -04001162 return ContainsUnconditionalBreak{}.visitStatement(stmt);
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001163}
1164
John Stiles92219b42020-06-15 12:32:24 -04001165static void move_all_but_break(std::unique_ptr<Statement>& stmt,
1166 std::vector<std::unique_ptr<Statement>>* target) {
Ethan Nicholase6592142020-09-08 10:22:09 -04001167 switch (stmt->kind()) {
1168 case Statement::Kind::kBlock: {
John Stiles92219b42020-06-15 12:32:24 -04001169 // Recurse into the block.
1170 Block& block = static_cast<Block&>(*stmt);
1171
1172 std::vector<std::unique_ptr<Statement>> blockStmts;
1173 blockStmts.reserve(block.fStatements.size());
1174 for (std::unique_ptr<Statement>& statementInBlock : block.fStatements) {
1175 move_all_but_break(statementInBlock, &blockStmts);
Ethan Nicholas739e1ca2020-06-11 12:16:14 -04001176 }
John Stiles92219b42020-06-15 12:32:24 -04001177
1178 target->push_back(std::make_unique<Block>(block.fOffset, std::move(blockStmts),
1179 block.fSymbols, block.fIsScope));
Ethan Nicholas739e1ca2020-06-11 12:16:14 -04001180 break;
John Stiles92219b42020-06-15 12:32:24 -04001181 }
1182
Ethan Nicholase6592142020-09-08 10:22:09 -04001183 case Statement::Kind::kBreak:
John Stiles92219b42020-06-15 12:32:24 -04001184 // Do not append a break to the target.
1185 break;
1186
Ethan Nicholas739e1ca2020-06-11 12:16:14 -04001187 default:
John Stiles92219b42020-06-15 12:32:24 -04001188 // Append normal statements to the target.
1189 target->push_back(std::move(stmt));
1190 break;
Ethan Nicholas739e1ca2020-06-11 12:16:14 -04001191 }
1192}
1193
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001194// Returns a block containing all of the statements that will be run if the given case matches
1195// (which, owing to the statements being owned by unique_ptrs, means the switch itself will be
1196// broken by this call and must then be discarded).
1197// Returns null (and leaves the switch unmodified) if no such simple reduction is possible, such as
1198// when break statements appear inside conditionals.
John Stiles92219b42020-06-15 12:32:24 -04001199static std::unique_ptr<Statement> block_for_case(SwitchStatement* switchStatement,
1200 SwitchCase* caseToCapture) {
1201 // We have to be careful to not move any of the pointers until after we're sure we're going to
1202 // succeed, so before we make any changes at all, we check the switch-cases to decide on a plan
1203 // of action. First, find the switch-case we are interested in.
1204 auto iter = switchStatement->fCases.begin();
1205 for (; iter != switchStatement->fCases.end(); ++iter) {
1206 if (iter->get() == caseToCapture) {
1207 break;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001208 }
John Stiles92219b42020-06-15 12:32:24 -04001209 }
1210
1211 // Next, walk forward through the rest of the switch. If we find a conditional break, we're
1212 // stuck and can't simplify at all. If we find an unconditional break, we have a range of
1213 // statements that we can use for simplification.
1214 auto startIter = iter;
1215 Statement* unconditionalBreakStmt = nullptr;
1216 for (; iter != switchStatement->fCases.end(); ++iter) {
1217 for (std::unique_ptr<Statement>& stmt : (*iter)->fStatements) {
1218 if (contains_conditional_break(*stmt)) {
1219 // We can't reduce switch-cases to a block when they have conditional breaks.
1220 return nullptr;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001221 }
John Stiles92219b42020-06-15 12:32:24 -04001222
1223 if (contains_unconditional_break(*stmt)) {
1224 // We found an unconditional break. We can use this block, but we need to strip
1225 // out the break statement.
1226 unconditionalBreakStmt = stmt.get();
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001227 break;
1228 }
1229 }
John Stiles92219b42020-06-15 12:32:24 -04001230
1231 if (unconditionalBreakStmt != nullptr) {
1232 break;
1233 }
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001234 }
John Stiles92219b42020-06-15 12:32:24 -04001235
1236 // We fell off the bottom of the switch or encountered a break. We know the range of statements
1237 // that we need to move over, and we know it's safe to do so.
1238 std::vector<std::unique_ptr<Statement>> caseStmts;
1239
1240 // We can move over most of the statements as-is.
1241 while (startIter != iter) {
1242 for (std::unique_ptr<Statement>& stmt : (*startIter)->fStatements) {
1243 caseStmts.push_back(std::move(stmt));
1244 }
1245 ++startIter;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001246 }
John Stiles92219b42020-06-15 12:32:24 -04001247
1248 // If we found an unconditional break at the end, we need to move what we can while avoiding
1249 // that break.
1250 if (unconditionalBreakStmt != nullptr) {
1251 for (std::unique_ptr<Statement>& stmt : (*startIter)->fStatements) {
1252 if (stmt.get() == unconditionalBreakStmt) {
1253 move_all_but_break(stmt, &caseStmts);
1254 unconditionalBreakStmt = nullptr;
1255 break;
1256 }
1257
1258 caseStmts.push_back(std::move(stmt));
1259 }
1260 }
1261
1262 SkASSERT(unconditionalBreakStmt == nullptr); // Verify that we fixed the unconditional break.
1263
1264 // Return our newly-synthesized block.
1265 return std::make_unique<Block>(/*offset=*/-1, std::move(caseStmts), switchStatement->fSymbols);
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001266}
1267
Ethan Nicholascb670962017-04-20 19:31:52 -04001268void Compiler::simplifyStatement(DefinitionMap& definitions,
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001269 BasicBlock& b,
1270 std::vector<BasicBlock::Node>::iterator* iter,
1271 std::unordered_set<const Variable*>* undefinedVariables,
1272 bool* outUpdated,
1273 bool* outNeedsRescan) {
Ethan Nicholascb670962017-04-20 19:31:52 -04001274 Statement* stmt = (*iter)->statement()->get();
Ethan Nicholase6592142020-09-08 10:22:09 -04001275 switch (stmt->kind()) {
1276 case Statement::Kind::kVarDeclaration: {
John Stilesa5a97b42020-08-18 11:19:07 -04001277 const auto& varDecl = stmt->as<VarDeclaration>();
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001278 if (varDecl.fVar->dead() &&
1279 (!varDecl.fValue ||
1280 !varDecl.fValue->hasSideEffects())) {
1281 if (varDecl.fValue) {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001282 SkASSERT((*iter)->statement()->get() == stmt);
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001283 if (!b.tryRemoveExpressionBefore(iter, varDecl.fValue.get())) {
1284 *outNeedsRescan = true;
Ethan Nicholascb670962017-04-20 19:31:52 -04001285 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001286 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001287 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001288 *outUpdated = true;
Ethan Nicholascb670962017-04-20 19:31:52 -04001289 }
1290 break;
1291 }
Ethan Nicholase6592142020-09-08 10:22:09 -04001292 case Statement::Kind::kIf: {
John Stilesa5a97b42020-08-18 11:19:07 -04001293 IfStatement& i = stmt->as<IfStatement>();
Ethan Nicholase6592142020-09-08 10:22:09 -04001294 if (i.fTest->kind() == Expression::Kind::kBoolLiteral) {
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001295 // constant if, collapse down to a single branch
John Stilesa5a97b42020-08-18 11:19:07 -04001296 if (i.fTest->as<BoolLiteral>().fValue) {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001297 SkASSERT(i.fIfTrue);
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001298 (*iter)->setStatement(std::move(i.fIfTrue));
1299 } else {
1300 if (i.fIfFalse) {
1301 (*iter)->setStatement(std::move(i.fIfFalse));
1302 } else {
1303 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
1304 }
1305 }
1306 *outUpdated = true;
1307 *outNeedsRescan = true;
1308 break;
1309 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001310 if (i.fIfFalse && i.fIfFalse->isEmpty()) {
1311 // else block doesn't do anything, remove it
1312 i.fIfFalse.reset();
1313 *outUpdated = true;
1314 *outNeedsRescan = true;
1315 }
1316 if (!i.fIfFalse && i.fIfTrue->isEmpty()) {
1317 // if block doesn't do anything, no else block
1318 if (i.fTest->hasSideEffects()) {
1319 // test has side effects, keep it
1320 (*iter)->setStatement(std::unique_ptr<Statement>(
1321 new ExpressionStatement(std::move(i.fTest))));
1322 } else {
1323 // no if, no else, no test side effects, kill the whole if
1324 // statement
1325 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
1326 }
1327 *outUpdated = true;
1328 *outNeedsRescan = true;
1329 }
1330 break;
1331 }
Ethan Nicholase6592142020-09-08 10:22:09 -04001332 case Statement::Kind::kSwitch: {
John Stilesa5a97b42020-08-18 11:19:07 -04001333 SwitchStatement& s = stmt->as<SwitchStatement>();
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001334 int64_t switchValue;
1335 if (fIRGenerator->getConstantInt(*s.fValue, &switchValue)) {
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001336 // switch is constant, replace it with the case that matches
1337 bool found = false;
1338 SwitchCase* defaultCase = nullptr;
John Stiles9d944232020-08-19 09:56:49 -04001339 for (const std::unique_ptr<SwitchCase>& c : s.fCases) {
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001340 if (!c->fValue) {
1341 defaultCase = c.get();
1342 continue;
1343 }
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001344 int64_t caseValue;
1345 SkAssertResult(fIRGenerator->getConstantInt(*c->fValue, &caseValue));
1346 if (caseValue == switchValue) {
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001347 std::unique_ptr<Statement> newBlock = block_for_case(&s, c.get());
1348 if (newBlock) {
1349 (*iter)->setStatement(std::move(newBlock));
John Stiles9d944232020-08-19 09:56:49 -04001350 found = true;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001351 break;
1352 } else {
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -04001353 if (s.fIsStatic && !(fFlags & kPermitInvalidStaticTests_Flag)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001354 this->error(s.fOffset,
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001355 "static switch contains non-static conditional break");
1356 s.fIsStatic = false;
1357 }
1358 return; // can't simplify
1359 }
1360 }
1361 }
1362 if (!found) {
1363 // no matching case. use default if it exists, or kill the whole thing
1364 if (defaultCase) {
1365 std::unique_ptr<Statement> newBlock = block_for_case(&s, defaultCase);
1366 if (newBlock) {
1367 (*iter)->setStatement(std::move(newBlock));
1368 } else {
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -04001369 if (s.fIsStatic && !(fFlags & kPermitInvalidStaticTests_Flag)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001370 this->error(s.fOffset,
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001371 "static switch contains non-static conditional break");
1372 s.fIsStatic = false;
1373 }
1374 return; // can't simplify
1375 }
1376 } else {
1377 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
1378 }
1379 }
1380 *outUpdated = true;
1381 *outNeedsRescan = true;
1382 }
1383 break;
1384 }
Ethan Nicholase6592142020-09-08 10:22:09 -04001385 case Statement::Kind::kExpression: {
John Stilesa5a97b42020-08-18 11:19:07 -04001386 ExpressionStatement& e = stmt->as<ExpressionStatement>();
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001387 SkASSERT((*iter)->statement()->get() == &e);
Ethan Nicholascb670962017-04-20 19:31:52 -04001388 if (!e.fExpression->hasSideEffects()) {
1389 // Expression statement with no side effects, kill it
1390 if (!b.tryRemoveExpressionBefore(iter, e.fExpression.get())) {
1391 *outNeedsRescan = true;
1392 }
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001393 SkASSERT((*iter)->statement()->get() == stmt);
Ethan Nicholascb670962017-04-20 19:31:52 -04001394 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
1395 *outUpdated = true;
1396 }
1397 break;
1398 }
1399 default:
1400 break;
1401 }
1402}
1403
John Stiles0cc193a2020-09-09 09:39:34 -04001404bool Compiler::scanCFG(FunctionDefinition& f) {
1405 bool madeChanges = false;
1406
Ethan Nicholascb670962017-04-20 19:31:52 -04001407 CFG cfg = CFGGenerator().getCFG(f);
1408 this->computeDataFlow(&cfg);
ethannicholas22f939e2016-10-13 13:25:34 -07001409
1410 // check for unreachable code
1411 for (size_t i = 0; i < cfg.fBlocks.size(); i++) {
John Stiles0cc193a2020-09-09 09:39:34 -04001412 const BasicBlock& block = cfg.fBlocks[i];
1413 if (i != cfg.fStart && !block.fEntrances.size() && block.fNodes.size()) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001414 int offset;
John Stiles0cc193a2020-09-09 09:39:34 -04001415 const BasicBlock::Node& node = block.fNodes[0];
1416 switch (node.fKind) {
Ethan Nicholas86a43402017-01-19 13:32:00 -05001417 case BasicBlock::Node::kStatement_Kind:
John Stiles0cc193a2020-09-09 09:39:34 -04001418 offset = (*node.statement())->fOffset;
Ethan Nicholas86a43402017-01-19 13:32:00 -05001419 break;
1420 case BasicBlock::Node::kExpression_Kind:
John Stiles0cc193a2020-09-09 09:39:34 -04001421 offset = (*node.expression())->fOffset;
1422 if ((*node.expression())->is<BoolLiteral>()) {
Ethan Nicholas70728ef2020-05-28 07:09:00 -04001423 // Function inlining can generate do { ... } while(false) loops which always
1424 // break, so the boolean condition is considered unreachable. Since not
1425 // being able to reach a literal is a non-issue in the first place, we
1426 // don't report an error in this case.
1427 continue;
1428 }
Ethan Nicholas86a43402017-01-19 13:32:00 -05001429 break;
1430 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001431 this->error(offset, String("unreachable"));
ethannicholas22f939e2016-10-13 13:25:34 -07001432 }
1433 }
1434 if (fErrorCount) {
John Stiles0cc193a2020-09-09 09:39:34 -04001435 return madeChanges;
ethannicholas22f939e2016-10-13 13:25:34 -07001436 }
1437
Ethan Nicholascb670962017-04-20 19:31:52 -04001438 // check for dead code & undefined variables, perform constant propagation
1439 std::unordered_set<const Variable*> undefinedVariables;
1440 bool updated;
1441 bool needsRescan = false;
1442 do {
1443 if (needsRescan) {
1444 cfg = CFGGenerator().getCFG(f);
1445 this->computeDataFlow(&cfg);
1446 needsRescan = false;
Ethan Nicholas113628d2017-02-02 16:11:39 -05001447 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001448
1449 updated = false;
Ethan Nicholas1de14812020-06-19 15:32:49 -04001450 bool first = true;
Ethan Nicholascb670962017-04-20 19:31:52 -04001451 for (BasicBlock& b : cfg.fBlocks) {
Ethan Nicholas1de14812020-06-19 15:32:49 -04001452 if (!first && b.fEntrances.empty()) {
1453 // Block was reachable before optimization, but has since become unreachable. In
1454 // addition to being dead code, it's broken - since control flow can't reach it, no
1455 // prior variable definitions can reach it, and therefore variables might look to
1456 // have not been properly assigned. Kill it.
Brian Osmandb16c482020-09-09 15:15:06 -04001457
1458 // We need to do this in two steps. For any variable declarations, the node list
1459 // will contain statement nodes for each VarDeclaration, and then a statement for
1460 // the VarDeclarationsStatement. When we replace the VDS with a Nop, we delete the
1461 // storage of the unique_ptr that the VD nodes are pointing to. So we remove those
1462 // from the node list entirely, first.
1463 b.fNodes.erase(
1464 std::remove_if(b.fNodes.begin(), b.fNodes.end(),
1465 [](const BasicBlock::Node& node) {
1466 return node.fKind == BasicBlock::Node::kStatement_Kind &&
1467 (*node.statement())->is<VarDeclaration>();
1468 }),
1469 b.fNodes.end());
1470
1471 // Now replace any remaining statements in the block with Nops.
Ethan Nicholas1de14812020-06-19 15:32:49 -04001472 for (BasicBlock::Node& node : b.fNodes) {
1473 if (node.fKind == BasicBlock::Node::kStatement_Kind &&
John Stiles0cc193a2020-09-09 09:39:34 -04001474 !(*node.statement())->is<Nop>()) {
1475 node.setStatement(std::make_unique<Nop>());
1476 madeChanges = true;
Ethan Nicholas1de14812020-06-19 15:32:49 -04001477 }
1478 }
1479 continue;
1480 }
1481 first = false;
Ethan Nicholascb670962017-04-20 19:31:52 -04001482 DefinitionMap definitions = b.fBefore;
1483
1484 for (auto iter = b.fNodes.begin(); iter != b.fNodes.end() && !needsRescan; ++iter) {
1485 if (iter->fKind == BasicBlock::Node::kExpression_Kind) {
1486 this->simplifyExpression(definitions, b, &iter, &undefinedVariables, &updated,
1487 &needsRescan);
1488 } else {
1489 this->simplifyStatement(definitions, b, &iter, &undefinedVariables, &updated,
John Stiles0cc193a2020-09-09 09:39:34 -04001490 &needsRescan);
Ethan Nicholascb670962017-04-20 19:31:52 -04001491 }
Ethan Nicholas4b330df2017-05-17 10:52:55 -04001492 if (needsRescan) {
1493 break;
1494 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001495 this->addDefinitions(*iter, &definitions);
1496 }
Brian Osman01a3eb42020-09-14 11:32:49 -04001497
1498 if (needsRescan) {
1499 break;
1500 }
Ethan Nicholascb670962017-04-20 19:31:52 -04001501 }
John Stiles0cc193a2020-09-09 09:39:34 -04001502 madeChanges |= updated;
Ethan Nicholascb670962017-04-20 19:31:52 -04001503 } while (updated);
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001504 SkASSERT(!needsRescan);
ethannicholas22f939e2016-10-13 13:25:34 -07001505
Ethan Nicholas91a10532017-06-22 11:24:38 -04001506 // verify static ifs & switches, clean up dead variable decls
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001507 for (BasicBlock& b : cfg.fBlocks) {
1508 DefinitionMap definitions = b.fBefore;
1509
Ethan Nicholas91a10532017-06-22 11:24:38 -04001510 for (auto iter = b.fNodes.begin(); iter != b.fNodes.end() && !needsRescan;) {
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001511 if (iter->fKind == BasicBlock::Node::kStatement_Kind) {
1512 const Statement& s = **iter->statement();
Ethan Nicholase6592142020-09-08 10:22:09 -04001513 switch (s.kind()) {
1514 case Statement::Kind::kIf:
John Stilesa5a97b42020-08-18 11:19:07 -04001515 if (s.as<IfStatement>().fIsStatic &&
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -04001516 !(fFlags & kPermitInvalidStaticTests_Flag)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001517 this->error(s.fOffset, "static if has non-static test");
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001518 }
Ethan Nicholas91a10532017-06-22 11:24:38 -04001519 ++iter;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001520 break;
Ethan Nicholase6592142020-09-08 10:22:09 -04001521 case Statement::Kind::kSwitch:
John Stilesa5a97b42020-08-18 11:19:07 -04001522 if (s.as<SwitchStatement>().fIsStatic &&
John Stiles0cc193a2020-09-09 09:39:34 -04001523 !(fFlags & kPermitInvalidStaticTests_Flag)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001524 this->error(s.fOffset, "static switch has non-static test");
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001525 }
Ethan Nicholas91a10532017-06-22 11:24:38 -04001526 ++iter;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001527 break;
Ethan Nicholase6592142020-09-08 10:22:09 -04001528 case Statement::Kind::kVarDeclarations: {
John Stilesa5a97b42020-08-18 11:19:07 -04001529 VarDeclarations& decls = *s.as<VarDeclarationsStatement>().fDeclaration;
John Stiles0cc193a2020-09-09 09:39:34 -04001530 decls.fVars.erase(
1531 std::remove_if(decls.fVars.begin(), decls.fVars.end(),
1532 [&](const std::unique_ptr<Statement>& var) {
1533 bool nop = var->is<Nop>();
1534 madeChanges |= nop;
1535 return nop;
1536 }),
1537 decls.fVars.end());
1538 if (decls.fVars.empty()) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001539 iter = b.fNodes.erase(iter);
1540 } else {
1541 ++iter;
1542 }
1543 break;
1544 }
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001545 default:
Ethan Nicholas91a10532017-06-22 11:24:38 -04001546 ++iter;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001547 break;
1548 }
Ethan Nicholas91a10532017-06-22 11:24:38 -04001549 } else {
1550 ++iter;
Ethan Nicholas5ac13c22017-05-10 15:06:17 -04001551 }
1552 }
1553 }
1554
ethannicholas22f939e2016-10-13 13:25:34 -07001555 // check for missing return
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001556 if (f.fDeclaration.fReturnType != *fContext->fVoid_Type) {
ethannicholas22f939e2016-10-13 13:25:34 -07001557 if (cfg.fBlocks[cfg.fExit].fEntrances.size()) {
Ethan Nicholasdb80f692019-11-22 14:06:12 -05001558 this->error(f.fOffset, String("function '" + String(f.fDeclaration.fName) +
1559 "' can exit without returning a value"));
ethannicholas22f939e2016-10-13 13:25:34 -07001560 }
1561 }
John Stiles0cc193a2020-09-09 09:39:34 -04001562
1563 return madeChanges;
ethannicholas22f939e2016-10-13 13:25:34 -07001564}
1565
Ethan Nicholas91164d12019-05-15 15:29:54 -04001566void Compiler::registerExternalValue(ExternalValue* value) {
1567 fIRGenerator->fRootSymbolTable->addWithoutOwnership(value->fName, value);
1568}
1569
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001570const Symbol* Compiler::takeOwnership(std::unique_ptr<const Symbol> symbol) {
John Stiles3ae071e2020-08-05 15:29:29 -04001571 return fIRGenerator->fRootSymbolTable->takeOwnershipOfSymbol(std::move(symbol));
Ethan Nicholas91164d12019-05-15 15:29:54 -04001572}
1573
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001574std::unique_ptr<Program> Compiler::convertProgram(Program::Kind kind, String text,
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001575 const Program::Settings& settings) {
ethannicholasb3058bd2016-07-01 08:22:01 -07001576 fErrorText = "";
1577 fErrorCount = 0;
John Stiles7b463002020-08-31 17:29:21 -04001578 fInliner.reset(context(), settings);
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001579 std::vector<std::unique_ptr<ProgramElement>>* inherited;
ethannicholasd598f792016-07-25 10:08:54 -07001580 std::vector<std::unique_ptr<ProgramElement>> elements;
ethannicholasb3058bd2016-07-01 08:22:01 -07001581 switch (kind) {
1582 case Program::kVertex_Kind:
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001583 inherited = &fVertexInclude;
1584 fIRGenerator->fSymbolTable = fVertexSymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001585 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001586 fIRGenerator->start(&settings, inherited);
ethannicholasb3058bd2016-07-01 08:22:01 -07001587 break;
1588 case Program::kFragment_Kind:
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001589 inherited = &fFragmentInclude;
1590 fIRGenerator->fSymbolTable = fFragmentSymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001591 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001592 fIRGenerator->start(&settings, inherited);
ethannicholasb3058bd2016-07-01 08:22:01 -07001593 break;
Ethan Nicholas52cad152017-02-16 16:37:32 -05001594 case Program::kGeometry_Kind:
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001595 this->loadGeometryIntrinsics();
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001596 inherited = &fGeometryInclude;
1597 fIRGenerator->fSymbolTable = fGeometrySymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001598 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001599 fIRGenerator->start(&settings, inherited);
Ethan Nicholas52cad152017-02-16 16:37:32 -05001600 break;
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -04001601 case Program::kFragmentProcessor_Kind: {
Brian Osmandd496172020-08-08 08:17:18 -04001602#if !SKSL_STANDALONE
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001603 {
1604 Rehydrator rehydrator(fContext.get(), fGpuSymbolTable, this,
1605 SKSL_INCLUDE_sksl_fp,
1606 SKSL_INCLUDE_sksl_fp_LENGTH);
1607 fFPSymbolTable = rehydrator.symbolTable();
1608 fFPInclude = rehydrator.elements();
1609 }
1610 inherited = &fFPInclude;
1611 fIRGenerator->fSymbolTable = fFPSymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001612 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001613 fIRGenerator->start(&settings, inherited);
1614 break;
1615#else
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001616 inherited = nullptr;
Ethan Nicholas8da1e652019-05-24 11:01:59 -04001617 fIRGenerator->fSymbolTable = fGpuSymbolTable;
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -04001618 fIRGenerator->start(&settings, /*inherited=*/nullptr, /*builtin=*/true);
John Stiles810c8cf2020-08-26 19:46:27 -04001619 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -04001620 std::ifstream in(SKSL_FP_INCLUDE);
1621 std::string stdText{std::istreambuf_iterator<char>(in),
1622 std::istreambuf_iterator<char>()};
1623 if (in.rdstate()) {
1624 printf("error reading %s\n", SKSL_FP_INCLUDE);
1625 abort();
1626 }
1627 const String* source = fGpuSymbolTable->takeOwnershipOfString(
1628 std::make_unique<String>(stdText.c_str()));
1629 fIRGenerator->convertProgram(kind, source->c_str(), source->length(), &elements);
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001630 fIRGenerator->fIsBuiltinCode = false;
Ethan Nicholas762466e2017-06-29 10:03:38 -04001631 break;
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001632#endif
Ethan Nicholasb33fa3f2020-08-06 13:00:19 -04001633 }
Ethan Nicholas8da1e652019-05-24 11:01:59 -04001634 case Program::kPipelineStage_Kind:
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001635 this->loadPipelineIntrinsics();
Ethan Nicholas8da1e652019-05-24 11:01:59 -04001636 inherited = &fPipelineInclude;
1637 fIRGenerator->fSymbolTable = fPipelineSymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001638 fIRGenerator->fIntrinsics = fGPUIntrinsics.get();
Ethan Nicholas8da1e652019-05-24 11:01:59 -04001639 fIRGenerator->start(&settings, inherited);
1640 break;
Ethan Nicholas746035a2019-04-23 13:31:09 -04001641 case Program::kGeneric_Kind:
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001642 this->loadInterpreterIntrinsics();
Brian Osmaneac49832020-09-18 11:49:22 -04001643 inherited = nullptr;
Ethan Nicholas8da1e652019-05-24 11:01:59 -04001644 fIRGenerator->fSymbolTable = fInterpreterSymbolTable;
John Stiles810c8cf2020-08-26 19:46:27 -04001645 fIRGenerator->fIntrinsics = fInterpreterIntrinsics.get();
Brian Osmaneac49832020-09-18 11:49:22 -04001646 fIRGenerator->start(&settings, /*inherited=*/nullptr);
Ethan Nicholas0d997662019-04-08 09:46:01 -04001647 break;
ethannicholasb3058bd2016-07-01 08:22:01 -07001648 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001649 std::unique_ptr<String> textPtr(new String(std::move(text)));
1650 fSource = textPtr.get();
Ethan Nicholasc18bb512020-07-28 14:46:53 -04001651 fIRGenerator->convertProgram(kind, textPtr->c_str(), textPtr->size(), &elements);
John Stilesfbd050b2020-08-03 13:21:46 -04001652 auto result = std::make_unique<Program>(kind,
1653 std::move(textPtr),
1654 settings,
1655 fContext,
1656 inherited,
1657 std::move(elements),
1658 fIRGenerator->fSymbolTable,
1659 fIRGenerator->fInputs);
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001660 if (fErrorCount) {
1661 return nullptr;
1662 }
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001663 if (settings.fOptimize && !this->optimize(*result)) {
1664 return nullptr;
1665 }
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001666 return result;
1667}
1668
Ethan Nicholas00543112018-07-31 09:44:36 -04001669bool Compiler::optimize(Program& program) {
1670 SkASSERT(!fErrorCount);
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001671 fIRGenerator->fKind = program.fKind;
1672 fIRGenerator->fSettings = &program.fSettings;
John Stiles7954d6c2020-09-01 10:53:02 -04001673
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001674 while (fErrorCount == 0) {
1675 bool madeChanges = false;
John Stiles7954d6c2020-09-01 10:53:02 -04001676
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001677 // Scan and optimize based on the control-flow graph for each function.
1678 for (ProgramElement& element : program) {
1679 if (element.is<FunctionDefinition>()) {
1680 madeChanges |= this->scanCFG(element.as<FunctionDefinition>());
1681 }
1682 }
1683
1684 // Perform inline-candidate analysis and inline any functions deemed suitable.
John Stiles910845f2020-09-18 15:41:07 -04001685 if (program.fSettings.fInline) {
1686 madeChanges |= fInliner.analyze(program);
1687 }
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001688
1689 // Remove dead functions. We wait until after analysis so that we still report errors,
1690 // even in unused code.
1691 if (program.fSettings.fRemoveDeadFunctions) {
1692 program.fElements.erase(
1693 std::remove_if(program.fElements.begin(),
1694 program.fElements.end(),
1695 [&](const std::unique_ptr<ProgramElement>& element) {
1696 if (!element->is<FunctionDefinition>()) {
1697 return false;
1698 }
1699 const auto& fn = element->as<FunctionDefinition>();
1700 bool dead = fn.fDeclaration.fCallCount == 0 &&
1701 fn.fDeclaration.fName != "main";
1702 madeChanges |= dead;
1703 return dead;
1704 }),
1705 program.fElements.end());
1706 }
1707
1708 if (program.fKind != Program::kFragmentProcessor_Kind) {
1709 // Remove dead variables.
John Stileseadfc3b2020-09-02 14:12:41 -04001710 for (ProgramElement& element : program) {
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001711 if (!element.is<VarDeclarations>()) {
1712 continue;
Ethan Nicholas0dc80872019-02-08 15:46:24 -05001713 }
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001714 VarDeclarations& vars = element.as<VarDeclarations>();
1715 vars.fVars.erase(
1716 std::remove_if(vars.fVars.begin(), vars.fVars.end(),
1717 [&](const std::unique_ptr<Statement>& stmt) {
1718 bool dead = stmt->as<VarDeclaration>().fVar->dead();
John Stiles73a6bff2020-09-09 13:40:37 -04001719 madeChanges |= dead;
1720 return dead;
1721 }),
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001722 vars.fVars.end());
John Stiles73a6bff2020-09-09 13:40:37 -04001723 }
1724
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001725 // Remove empty variable declarations with no variables left inside of them.
1726 program.fElements.erase(
1727 std::remove_if(program.fElements.begin(), program.fElements.end(),
1728 [&](const std::unique_ptr<ProgramElement>& element) {
1729 if (!element->is<VarDeclarations>()) {
1730 return false;
1731 }
1732 bool dead = element->as<VarDeclarations>().fVars.empty();
1733 madeChanges |= dead;
1734 return dead;
1735 }),
1736 program.fElements.end());
1737 }
John Stiles73a6bff2020-09-09 13:40:37 -04001738
Ethan Nicholas34b19c52020-09-14 11:33:47 -04001739 if (!madeChanges) {
1740 break;
Ethan Nicholas0dc80872019-02-08 15:46:24 -05001741 }
Ethan Nicholas00543112018-07-31 09:44:36 -04001742 }
1743 return fErrorCount == 0;
1744}
1745
Brian Osmanfb32ddf2019-06-18 10:14:20 -04001746#if defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
1747
Ethan Nicholas00543112018-07-31 09:44:36 -04001748bool Compiler::toSPIRV(Program& program, OutputStream& out) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001749#ifdef SK_ENABLE_SPIRV_VALIDATION
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001750 StringStream buffer;
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001751 fSource = program.fSource.get();
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001752 SPIRVCodeGenerator cg(fContext.get(), &program, this, &buffer);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001753 bool result = cg.generateCode();
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001754 fSource = nullptr;
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001755 if (result) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001756 spvtools::SpirvTools tools(SPV_ENV_VULKAN_1_0);
Ethan Nicholas762466e2017-06-29 10:03:38 -04001757 const String& data = buffer.str();
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001758 SkASSERT(0 == data.size() % 4);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001759 auto dumpmsg = [](spv_message_level_t, const char*, const spv_position_t&, const char* m) {
1760 SkDebugf("SPIR-V validation error: %s\n", m);
1761 };
1762 tools.SetMessageConsumer(dumpmsg);
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001763 // Verify that the SPIR-V we produced is valid. If this SkASSERT fails, check the logs prior
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001764 // to the failure to see the validation errors.
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001765 SkAssertResult(tools.Validate((const uint32_t*) data.c_str(), data.size() / 4));
Ethan Nicholas762466e2017-06-29 10:03:38 -04001766 out.write(data.c_str(), data.size());
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001767 }
1768#else
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001769 fSource = program.fSource.get();
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001770 SPIRVCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001771 bool result = cg.generateCode();
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001772 fSource = nullptr;
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -04001773#endif
Ethan Nicholasce33f102016-12-09 17:22:59 -05001774 return result;
1775}
1776
Ethan Nicholas00543112018-07-31 09:44:36 -04001777bool Compiler::toSPIRV(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001778 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001779 bool result = this->toSPIRV(program, buffer);
1780 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001781 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001782 }
1783 return result;
1784}
1785
Ethan Nicholas00543112018-07-31 09:44:36 -04001786bool Compiler::toGLSL(Program& program, OutputStream& out) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001787 fSource = program.fSource.get();
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001788 GLSLCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001789 bool result = cg.generateCode();
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001790 fSource = nullptr;
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001791 return result;
1792}
1793
Ethan Nicholas00543112018-07-31 09:44:36 -04001794bool Compiler::toGLSL(Program& program, String* out) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001795 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001796 bool result = this->toGLSL(program, buffer);
1797 if (result) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001798 *out = buffer.str();
Ethan Nicholas941e7e22016-12-12 15:33:30 -05001799 }
1800 return result;
1801}
1802
Brian Osmanc0243912020-02-19 15:35:26 -05001803bool Compiler::toHLSL(Program& program, String* out) {
1804 String spirv;
1805 if (!this->toSPIRV(program, &spirv)) {
1806 return false;
1807 }
1808
1809 return SPIRVtoHLSL(spirv, out);
1810}
1811
Ethan Nicholas00543112018-07-31 09:44:36 -04001812bool Compiler::toMetal(Program& program, OutputStream& out) {
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001813 MetalCodeGenerator cg(fContext.get(), &program, this, &out);
Ethan Nicholascc305772017-10-13 16:17:45 -04001814 bool result = cg.generateCode();
Ethan Nicholascc305772017-10-13 16:17:45 -04001815 return result;
1816}
1817
Ethan Nicholas00543112018-07-31 09:44:36 -04001818bool Compiler::toMetal(Program& program, String* out) {
Timothy Liangb8eeb802018-07-23 16:46:16 -04001819 StringStream buffer;
1820 bool result = this->toMetal(program, buffer);
1821 if (result) {
1822 *out = buffer.str();
1823 }
1824 return result;
1825}
1826
Ethan Nicholas2a479a52020-08-18 16:29:45 -04001827#if defined(SKSL_STANDALONE) || defined(GR_TEST_UTILS)
Ethan Nicholas00543112018-07-31 09:44:36 -04001828bool Compiler::toCPP(Program& program, String name, OutputStream& out) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001829 fSource = program.fSource.get();
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001830 CPPCodeGenerator cg(fContext.get(), &program, this, name, &out);
Ethan Nicholas762466e2017-06-29 10:03:38 -04001831 bool result = cg.generateCode();
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001832 fSource = nullptr;
Ethan Nicholas762466e2017-06-29 10:03:38 -04001833 return result;
1834}
1835
Ethan Nicholas00543112018-07-31 09:44:36 -04001836bool Compiler::toH(Program& program, String name, OutputStream& out) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001837 fSource = program.fSource.get();
Ethan Nicholas26a9aad2018-03-27 14:10:52 -04001838 HCodeGenerator cg(fContext.get(), &program, this, name, &out);
Ethan Nicholas762466e2017-06-29 10:03:38 -04001839 bool result = cg.generateCode();
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001840 fSource = nullptr;
Ethan Nicholas00543112018-07-31 09:44:36 -04001841 return result;
1842}
Ethan Nicholas2a479a52020-08-18 16:29:45 -04001843#endif // defined(SKSL_STANDALONE) || defined(GR_TEST_UTILS)
Ethan Nicholas00543112018-07-31 09:44:36 -04001844
Ethan Nicholas2a479a52020-08-18 16:29:45 -04001845#endif // defined(SKSL_STANDALONE) || SK_SUPPORT_GPU
Brian Osman2e29ab52019-09-20 12:19:11 -04001846
1847#if !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
Brian Osmana4b91692020-08-10 14:26:16 -04001848bool Compiler::toPipelineStage(Program& program, PipelineStageArgs* outArgs) {
Ethan Nicholas00543112018-07-31 09:44:36 -04001849 fSource = program.fSource.get();
1850 StringStream buffer;
Brian Osman300fe1d2020-01-23 15:42:43 -05001851 PipelineStageCodeGenerator cg(fContext.get(), &program, this, &buffer, outArgs);
Ethan Nicholas00543112018-07-31 09:44:36 -04001852 bool result = cg.generateCode();
1853 fSource = nullptr;
1854 if (result) {
Brian Osman107c6662019-12-30 15:02:30 -05001855 outArgs->fCode = buffer.str();
Ethan Nicholas00543112018-07-31 09:44:36 -04001856 }
Ethan Nicholas762466e2017-06-29 10:03:38 -04001857 return result;
1858}
Brian Osmanfb32ddf2019-06-18 10:14:20 -04001859#endif
1860
Ethan Nicholas0e9401d2019-03-21 11:05:37 -04001861std::unique_ptr<ByteCode> Compiler::toByteCode(Program& program) {
Mike Klein853d4ed2020-08-20 00:41:00 +00001862#if defined(SK_ENABLE_SKSL_INTERPRETER)
Brian Osman808f0212020-01-21 15:36:47 -05001863 fSource = program.fSource.get();
Ethan Nicholas0e9401d2019-03-21 11:05:37 -04001864 std::unique_ptr<ByteCode> result(new ByteCode());
Brian Osmanb08cc022020-04-02 11:38:40 -04001865 ByteCodeGenerator cg(fContext.get(), &program, this, result.get());
1866 bool success = cg.generateCode();
1867 fSource = nullptr;
1868 if (success) {
Ethan Nicholas0e9401d2019-03-21 11:05:37 -04001869 return result;
1870 }
Mike Klein853d4ed2020-08-20 00:41:00 +00001871#else
1872 ABORT("ByteCode interpreter not enabled");
1873#endif
Ethan Nicholas0e9401d2019-03-21 11:05:37 -04001874 return nullptr;
1875}
1876
Brian Osman401a0092020-09-10 14:47:24 -04001877const char* Compiler::OperatorName(Token::Kind op) {
1878 switch (op) {
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001879 case Token::Kind::TK_PLUS: return "+";
1880 case Token::Kind::TK_MINUS: return "-";
1881 case Token::Kind::TK_STAR: return "*";
1882 case Token::Kind::TK_SLASH: return "/";
1883 case Token::Kind::TK_PERCENT: return "%";
1884 case Token::Kind::TK_SHL: return "<<";
1885 case Token::Kind::TK_SHR: return ">>";
1886 case Token::Kind::TK_LOGICALNOT: return "!";
1887 case Token::Kind::TK_LOGICALAND: return "&&";
1888 case Token::Kind::TK_LOGICALOR: return "||";
1889 case Token::Kind::TK_LOGICALXOR: return "^^";
1890 case Token::Kind::TK_BITWISENOT: return "~";
1891 case Token::Kind::TK_BITWISEAND: return "&";
1892 case Token::Kind::TK_BITWISEOR: return "|";
1893 case Token::Kind::TK_BITWISEXOR: return "^";
1894 case Token::Kind::TK_EQ: return "=";
1895 case Token::Kind::TK_EQEQ: return "==";
1896 case Token::Kind::TK_NEQ: return "!=";
1897 case Token::Kind::TK_LT: return "<";
1898 case Token::Kind::TK_GT: return ">";
1899 case Token::Kind::TK_LTEQ: return "<=";
1900 case Token::Kind::TK_GTEQ: return ">=";
1901 case Token::Kind::TK_PLUSEQ: return "+=";
1902 case Token::Kind::TK_MINUSEQ: return "-=";
1903 case Token::Kind::TK_STAREQ: return "*=";
1904 case Token::Kind::TK_SLASHEQ: return "/=";
1905 case Token::Kind::TK_PERCENTEQ: return "%=";
1906 case Token::Kind::TK_SHLEQ: return "<<=";
1907 case Token::Kind::TK_SHREQ: return ">>=";
1908 case Token::Kind::TK_LOGICALANDEQ: return "&&=";
1909 case Token::Kind::TK_LOGICALOREQ: return "||=";
1910 case Token::Kind::TK_LOGICALXOREQ: return "^^=";
1911 case Token::Kind::TK_BITWISEANDEQ: return "&=";
1912 case Token::Kind::TK_BITWISEOREQ: return "|=";
1913 case Token::Kind::TK_BITWISEXOREQ: return "^=";
1914 case Token::Kind::TK_PLUSPLUS: return "++";
1915 case Token::Kind::TK_MINUSMINUS: return "--";
1916 case Token::Kind::TK_COMMA: return ",";
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001917 default:
Brian Osman401a0092020-09-10 14:47:24 -04001918 ABORT("unsupported operator: %d\n", (int) op);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001919 }
1920}
1921
1922
1923bool Compiler::IsAssignment(Token::Kind op) {
1924 switch (op) {
Ethan Nicholas5a9e7fb2020-04-17 12:45:51 -04001925 case Token::Kind::TK_EQ: // fall through
1926 case Token::Kind::TK_PLUSEQ: // fall through
1927 case Token::Kind::TK_MINUSEQ: // fall through
1928 case Token::Kind::TK_STAREQ: // fall through
1929 case Token::Kind::TK_SLASHEQ: // fall through
1930 case Token::Kind::TK_PERCENTEQ: // fall through
1931 case Token::Kind::TK_SHLEQ: // fall through
1932 case Token::Kind::TK_SHREQ: // fall through
1933 case Token::Kind::TK_BITWISEOREQ: // fall through
1934 case Token::Kind::TK_BITWISEXOREQ: // fall through
1935 case Token::Kind::TK_BITWISEANDEQ: // fall through
1936 case Token::Kind::TK_LOGICALOREQ: // fall through
1937 case Token::Kind::TK_LOGICALXOREQ: // fall through
1938 case Token::Kind::TK_LOGICALANDEQ:
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001939 return true;
1940 default:
1941 return false;
1942 }
1943}
1944
Brian Osman401a0092020-09-10 14:47:24 -04001945Token::Kind Compiler::RemoveAssignment(Token::Kind op) {
1946 switch (op) {
1947 case Token::Kind::TK_PLUSEQ: return Token::Kind::TK_PLUS;
1948 case Token::Kind::TK_MINUSEQ: return Token::Kind::TK_MINUS;
1949 case Token::Kind::TK_STAREQ: return Token::Kind::TK_STAR;
1950 case Token::Kind::TK_SLASHEQ: return Token::Kind::TK_SLASH;
1951 case Token::Kind::TK_PERCENTEQ: return Token::Kind::TK_PERCENT;
1952 case Token::Kind::TK_SHLEQ: return Token::Kind::TK_SHL;
1953 case Token::Kind::TK_SHREQ: return Token::Kind::TK_SHR;
1954 case Token::Kind::TK_BITWISEOREQ: return Token::Kind::TK_BITWISEOR;
1955 case Token::Kind::TK_BITWISEXOREQ: return Token::Kind::TK_BITWISEXOR;
1956 case Token::Kind::TK_BITWISEANDEQ: return Token::Kind::TK_BITWISEAND;
1957 case Token::Kind::TK_LOGICALOREQ: return Token::Kind::TK_LOGICALOR;
1958 case Token::Kind::TK_LOGICALXOREQ: return Token::Kind::TK_LOGICALXOR;
1959 case Token::Kind::TK_LOGICALANDEQ: return Token::Kind::TK_LOGICALAND;
1960 default: return op;
1961 }
1962}
1963
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001964Position Compiler::position(int offset) {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -04001965 SkASSERT(fSource);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001966 int line = 1;
1967 int column = 1;
1968 for (int i = 0; i < offset; i++) {
1969 if ((*fSource)[i] == '\n') {
1970 ++line;
1971 column = 1;
1972 }
1973 else {
1974 ++column;
1975 }
1976 }
1977 return Position(line, column);
1978}
1979
1980void Compiler::error(int offset, String msg) {
ethannicholasb3058bd2016-07-01 08:22:01 -07001981 fErrorCount++;
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001982 Position pos = this->position(offset);
1983 fErrorText += "error: " + to_string(pos.fLine) + ": " + msg.c_str() + "\n";
ethannicholasb3058bd2016-07-01 08:22:01 -07001984}
1985
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001986String Compiler::errorText() {
Ethan Nicholas00543112018-07-31 09:44:36 -04001987 this->writeErrorCount();
1988 fErrorCount = 0;
Ethan Nicholas0df1b042017-03-31 13:56:23 -04001989 String result = fErrorText;
ethannicholasb3058bd2016-07-01 08:22:01 -07001990 return result;
1991}
1992
1993void Compiler::writeErrorCount() {
1994 if (fErrorCount) {
1995 fErrorText += to_string(fErrorCount) + " error";
1996 if (fErrorCount > 1) {
1997 fErrorText += "s";
1998 }
1999 fErrorText += "\n";
2000 }
2001}
2002
John Stilesa6841be2020-08-06 14:11:56 -04002003} // namespace SkSL