2016-07-01 15:22:01 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
2016-10-26 14:35:22 +00:00
|
|
|
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "SkSLCompiler.h"
|
|
|
|
|
2019-03-21 15:05:37 +00:00
|
|
|
#include "SkSLByteCodeGenerator.h"
|
2016-10-13 20:25:34 +00:00
|
|
|
#include "SkSLCFGGenerator.h"
|
2017-06-29 14:03:38 +00:00
|
|
|
#include "SkSLCPPCodeGenerator.h"
|
2016-12-12 20:33:30 +00:00
|
|
|
#include "SkSLGLSLCodeGenerator.h"
|
2017-06-29 14:03:38 +00:00
|
|
|
#include "SkSLHCodeGenerator.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "SkSLIRGenerator.h"
|
2017-10-13 20:17:45 +00:00
|
|
|
#include "SkSLMetalCodeGenerator.h"
|
2018-07-31 13:44:36 +00:00
|
|
|
#include "SkSLPipelineStageCodeGenerator.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "SkSLSPIRVCodeGenerator.h"
|
2017-11-10 20:34:03 +00:00
|
|
|
#include "ir/SkSLEnum.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "ir/SkSLExpression.h"
|
2017-04-20 23:31:52 +00:00
|
|
|
#include "ir/SkSLExpressionStatement.h"
|
2018-03-29 20:46:56 +00:00
|
|
|
#include "ir/SkSLFunctionCall.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "ir/SkSLIntLiteral.h"
|
2016-10-12 13:39:56 +00:00
|
|
|
#include "ir/SkSLModifiersDeclaration.h"
|
2017-04-20 23:31:52 +00:00
|
|
|
#include "ir/SkSLNop.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
#include "ir/SkSLSymbolTable.h"
|
2017-04-20 23:31:52 +00:00
|
|
|
#include "ir/SkSLTernaryExpression.h"
|
2016-10-20 16:54:00 +00:00
|
|
|
#include "ir/SkSLUnresolvedFunction.h"
|
2016-10-13 20:25:34 +00:00
|
|
|
#include "ir/SkSLVarDeclarations.h"
|
2016-07-01 15:22:01 +00:00
|
|
|
|
2017-03-16 13:56:54 +00:00
|
|
|
#ifdef SK_ENABLE_SPIRV_VALIDATION
|
|
|
|
#include "spirv-tools/libspirv.hpp"
|
|
|
|
#endif
|
|
|
|
|
2016-07-01 15:22:01 +00:00
|
|
|
// include the built-in shader symbols as static strings
|
|
|
|
|
2017-11-16 16:20:11 +00:00
|
|
|
#define STRINGIFY(x) #x
|
|
|
|
|
2016-10-12 13:39:56 +00:00
|
|
|
static const char* SKSL_INCLUDE =
|
2018-01-24 22:32:17 +00:00
|
|
|
#include "sksl.inc"
|
2016-07-01 15:22:01 +00:00
|
|
|
;
|
|
|
|
|
2016-10-12 13:39:56 +00:00
|
|
|
static const char* SKSL_VERT_INCLUDE =
|
2018-01-24 22:32:17 +00:00
|
|
|
#include "sksl_vert.inc"
|
2016-07-01 15:22:01 +00:00
|
|
|
;
|
|
|
|
|
2016-10-12 13:39:56 +00:00
|
|
|
static const char* SKSL_FRAG_INCLUDE =
|
2018-01-24 22:32:17 +00:00
|
|
|
#include "sksl_frag.inc"
|
2016-07-01 15:22:01 +00:00
|
|
|
;
|
|
|
|
|
2017-02-16 21:37:32 +00:00
|
|
|
static const char* SKSL_GEOM_INCLUDE =
|
2018-01-24 22:32:17 +00:00
|
|
|
#include "sksl_geom.inc"
|
2017-02-16 21:37:32 +00:00
|
|
|
;
|
|
|
|
|
2017-06-29 14:03:38 +00:00
|
|
|
static const char* SKSL_FP_INCLUDE =
|
2018-01-24 22:32:17 +00:00
|
|
|
#include "sksl_enums.inc"
|
|
|
|
#include "sksl_fp.inc"
|
2017-06-29 14:03:38 +00:00
|
|
|
;
|
|
|
|
|
2019-04-23 17:31:09 +00:00
|
|
|
static const char* SKSL_GENERIC_INCLUDE =
|
|
|
|
#include "sksl_generic.inc"
|
2019-04-08 13:46:01 +00:00
|
|
|
;
|
|
|
|
|
2016-07-01 15:22:01 +00:00
|
|
|
namespace SkSL {
|
|
|
|
|
2017-07-14 14:12:15 +00:00
|
|
|
Compiler::Compiler(Flags flags)
|
|
|
|
: fFlags(flags)
|
2018-03-27 18:10:52 +00:00
|
|
|
, fContext(new Context())
|
2017-07-14 14:12:15 +00:00
|
|
|
, fErrorCount(0) {
|
2017-03-30 18:11:58 +00:00
|
|
|
auto types = std::shared_ptr<SymbolTable>(new SymbolTable(this));
|
|
|
|
auto symbols = std::shared_ptr<SymbolTable>(new SymbolTable(types, this));
|
2018-03-27 18:10:52 +00:00
|
|
|
fIRGenerator = new IRGenerator(fContext.get(), symbols, *this);
|
2016-07-01 15:22:01 +00:00
|
|
|
fTypes = types;
|
2018-03-27 18:10:52 +00:00
|
|
|
#define ADD_TYPE(t) types->addWithoutOwnership(fContext->f ## t ## _Type->fName, \
|
|
|
|
fContext->f ## t ## _Type.get())
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(Void);
|
|
|
|
ADD_TYPE(Float);
|
2017-07-28 19:19:46 +00:00
|
|
|
ADD_TYPE(Float2);
|
|
|
|
ADD_TYPE(Float3);
|
|
|
|
ADD_TYPE(Float4);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(Half);
|
|
|
|
ADD_TYPE(Half2);
|
|
|
|
ADD_TYPE(Half3);
|
|
|
|
ADD_TYPE(Half4);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(Double);
|
2017-07-28 19:19:46 +00:00
|
|
|
ADD_TYPE(Double2);
|
|
|
|
ADD_TYPE(Double3);
|
|
|
|
ADD_TYPE(Double4);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(Int);
|
2017-07-28 19:19:46 +00:00
|
|
|
ADD_TYPE(Int2);
|
|
|
|
ADD_TYPE(Int3);
|
|
|
|
ADD_TYPE(Int4);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(UInt);
|
2017-07-28 19:19:46 +00:00
|
|
|
ADD_TYPE(UInt2);
|
|
|
|
ADD_TYPE(UInt3);
|
|
|
|
ADD_TYPE(UInt4);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(Short);
|
|
|
|
ADD_TYPE(Short2);
|
|
|
|
ADD_TYPE(Short3);
|
|
|
|
ADD_TYPE(Short4);
|
|
|
|
ADD_TYPE(UShort);
|
|
|
|
ADD_TYPE(UShort2);
|
|
|
|
ADD_TYPE(UShort3);
|
|
|
|
ADD_TYPE(UShort4);
|
2018-07-17 14:19:38 +00:00
|
|
|
ADD_TYPE(Byte);
|
|
|
|
ADD_TYPE(Byte2);
|
|
|
|
ADD_TYPE(Byte3);
|
|
|
|
ADD_TYPE(Byte4);
|
|
|
|
ADD_TYPE(UByte);
|
|
|
|
ADD_TYPE(UByte2);
|
|
|
|
ADD_TYPE(UByte3);
|
|
|
|
ADD_TYPE(UByte4);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(Bool);
|
2017-07-28 19:19:46 +00:00
|
|
|
ADD_TYPE(Bool2);
|
|
|
|
ADD_TYPE(Bool3);
|
|
|
|
ADD_TYPE(Bool4);
|
|
|
|
ADD_TYPE(Float2x2);
|
|
|
|
ADD_TYPE(Float2x3);
|
|
|
|
ADD_TYPE(Float2x4);
|
|
|
|
ADD_TYPE(Float3x2);
|
|
|
|
ADD_TYPE(Float3x3);
|
|
|
|
ADD_TYPE(Float3x4);
|
|
|
|
ADD_TYPE(Float4x2);
|
|
|
|
ADD_TYPE(Float4x3);
|
|
|
|
ADD_TYPE(Float4x4);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(Half2x2);
|
|
|
|
ADD_TYPE(Half2x3);
|
|
|
|
ADD_TYPE(Half2x4);
|
|
|
|
ADD_TYPE(Half3x2);
|
|
|
|
ADD_TYPE(Half3x3);
|
|
|
|
ADD_TYPE(Half3x4);
|
|
|
|
ADD_TYPE(Half4x2);
|
|
|
|
ADD_TYPE(Half4x3);
|
|
|
|
ADD_TYPE(Half4x4);
|
|
|
|
ADD_TYPE(Double2x2);
|
|
|
|
ADD_TYPE(Double2x3);
|
|
|
|
ADD_TYPE(Double2x4);
|
|
|
|
ADD_TYPE(Double3x2);
|
|
|
|
ADD_TYPE(Double3x3);
|
|
|
|
ADD_TYPE(Double3x4);
|
|
|
|
ADD_TYPE(Double4x2);
|
|
|
|
ADD_TYPE(Double4x3);
|
|
|
|
ADD_TYPE(Double4x4);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(GenType);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(GenHType);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(GenDType);
|
|
|
|
ADD_TYPE(GenIType);
|
|
|
|
ADD_TYPE(GenUType);
|
|
|
|
ADD_TYPE(GenBType);
|
|
|
|
ADD_TYPE(Mat);
|
|
|
|
ADD_TYPE(Vec);
|
|
|
|
ADD_TYPE(GVec);
|
|
|
|
ADD_TYPE(GVec2);
|
|
|
|
ADD_TYPE(GVec3);
|
|
|
|
ADD_TYPE(GVec4);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(HVec);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(DVec);
|
|
|
|
ADD_TYPE(IVec);
|
|
|
|
ADD_TYPE(UVec);
|
2017-08-02 14:52:54 +00:00
|
|
|
ADD_TYPE(SVec);
|
|
|
|
ADD_TYPE(USVec);
|
2018-07-17 14:19:38 +00:00
|
|
|
ADD_TYPE(ByteVec);
|
|
|
|
ADD_TYPE(UByteVec);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(BVec);
|
|
|
|
|
|
|
|
ADD_TYPE(Sampler1D);
|
|
|
|
ADD_TYPE(Sampler2D);
|
|
|
|
ADD_TYPE(Sampler3D);
|
2016-10-12 13:39:56 +00:00
|
|
|
ADD_TYPE(SamplerExternalOES);
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(SamplerCube);
|
|
|
|
ADD_TYPE(Sampler2DRect);
|
|
|
|
ADD_TYPE(Sampler1DArray);
|
|
|
|
ADD_TYPE(Sampler2DArray);
|
|
|
|
ADD_TYPE(SamplerCubeArray);
|
|
|
|
ADD_TYPE(SamplerBuffer);
|
|
|
|
ADD_TYPE(Sampler2DMS);
|
|
|
|
ADD_TYPE(Sampler2DMSArray);
|
|
|
|
|
2016-11-11 21:08:03 +00:00
|
|
|
ADD_TYPE(ISampler2D);
|
|
|
|
|
2016-11-16 17:06:01 +00:00
|
|
|
ADD_TYPE(Image2D);
|
|
|
|
ADD_TYPE(IImage2D);
|
|
|
|
|
2016-11-22 14:44:03 +00:00
|
|
|
ADD_TYPE(SubpassInput);
|
|
|
|
ADD_TYPE(SubpassInputMS);
|
|
|
|
|
2016-07-01 15:22:01 +00:00
|
|
|
ADD_TYPE(GSampler1D);
|
|
|
|
ADD_TYPE(GSampler2D);
|
|
|
|
ADD_TYPE(GSampler3D);
|
|
|
|
ADD_TYPE(GSamplerCube);
|
|
|
|
ADD_TYPE(GSampler2DRect);
|
|
|
|
ADD_TYPE(GSampler1DArray);
|
|
|
|
ADD_TYPE(GSampler2DArray);
|
|
|
|
ADD_TYPE(GSamplerCubeArray);
|
|
|
|
ADD_TYPE(GSamplerBuffer);
|
|
|
|
ADD_TYPE(GSampler2DMS);
|
|
|
|
ADD_TYPE(GSampler2DMSArray);
|
|
|
|
|
|
|
|
ADD_TYPE(Sampler1DShadow);
|
|
|
|
ADD_TYPE(Sampler2DShadow);
|
|
|
|
ADD_TYPE(SamplerCubeShadow);
|
|
|
|
ADD_TYPE(Sampler2DRectShadow);
|
|
|
|
ADD_TYPE(Sampler1DArrayShadow);
|
|
|
|
ADD_TYPE(Sampler2DArrayShadow);
|
|
|
|
ADD_TYPE(SamplerCubeArrayShadow);
|
|
|
|
ADD_TYPE(GSampler2DArrayShadow);
|
|
|
|
ADD_TYPE(GSamplerCubeArrayShadow);
|
2017-10-10 20:30:21 +00:00
|
|
|
ADD_TYPE(FragmentProcessor);
|
2018-03-27 18:10:52 +00:00
|
|
|
ADD_TYPE(SkRasterPipeline);
|
2016-07-01 15:22:01 +00:00
|
|
|
|
2017-09-11 20:50:14 +00:00
|
|
|
StringFragment skCapsName("sk_Caps");
|
|
|
|
Variable* skCaps = new Variable(-1, Modifiers(), skCapsName,
|
2018-03-27 18:10:52 +00:00
|
|
|
*fContext->fSkCaps_Type, Variable::kGlobal_Storage);
|
2016-11-21 20:59:48 +00:00
|
|
|
fIRGenerator->fSymbolTable->add(skCapsName, std::unique_ptr<Symbol>(skCaps));
|
|
|
|
|
2017-09-11 20:50:14 +00:00
|
|
|
StringFragment skArgsName("sk_Args");
|
|
|
|
Variable* skArgs = new Variable(-1, Modifiers(), skArgsName,
|
2018-03-27 18:10:52 +00:00
|
|
|
*fContext->fSkArgs_Type, Variable::kGlobal_Storage);
|
2017-06-29 14:03:38 +00:00
|
|
|
fIRGenerator->fSymbolTable->add(skArgsName, std::unique_ptr<Symbol>(skArgs));
|
|
|
|
|
2017-09-18 18:10:39 +00:00
|
|
|
std::vector<std::unique_ptr<ProgramElement>> ignored;
|
2018-01-24 14:52:02 +00:00
|
|
|
fIRGenerator->convertProgram(Program::kFragment_Kind, SKSL_INCLUDE, strlen(SKSL_INCLUDE),
|
|
|
|
*fTypes, &ignored);
|
2016-10-20 16:54:00 +00:00
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
2017-09-11 20:50:14 +00:00
|
|
|
if (fErrorCount) {
|
|
|
|
printf("Unexpected errors: %s\n", fErrorText.c_str());
|
|
|
|
}
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(!fErrorCount);
|
2018-04-24 17:06:09 +00:00
|
|
|
|
|
|
|
Program::Settings settings;
|
|
|
|
fIRGenerator->start(&settings, nullptr);
|
|
|
|
fIRGenerator->convertProgram(Program::kFragment_Kind, SKSL_VERT_INCLUDE,
|
|
|
|
strlen(SKSL_VERT_INCLUDE), *fTypes, &fVertexInclude);
|
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
|
|
|
fVertexSymbolTable = fIRGenerator->fSymbolTable;
|
|
|
|
|
|
|
|
fIRGenerator->start(&settings, nullptr);
|
|
|
|
fIRGenerator->convertProgram(Program::kVertex_Kind, SKSL_FRAG_INCLUDE,
|
|
|
|
strlen(SKSL_FRAG_INCLUDE), *fTypes, &fFragmentInclude);
|
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
|
|
|
fFragmentSymbolTable = fIRGenerator->fSymbolTable;
|
|
|
|
|
|
|
|
fIRGenerator->start(&settings, nullptr);
|
|
|
|
fIRGenerator->convertProgram(Program::kGeometry_Kind, SKSL_GEOM_INCLUDE,
|
|
|
|
strlen(SKSL_GEOM_INCLUDE), *fTypes, &fGeometryInclude);
|
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
|
|
|
fGeometrySymbolTable = fIRGenerator->fSymbolTable;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Compiler::~Compiler() {
|
|
|
|
delete fIRGenerator;
|
|
|
|
}
|
|
|
|
|
2016-10-13 20:25:34 +00:00
|
|
|
// add the definition created by assigning to the lvalue to the definition set
|
2017-01-19 18:32:00 +00:00
|
|
|
void Compiler::addDefinition(const Expression* lvalue, std::unique_ptr<Expression>* expr,
|
|
|
|
DefinitionMap* definitions) {
|
2016-10-13 20:25:34 +00:00
|
|
|
switch (lvalue->fKind) {
|
|
|
|
case Expression::kVariableReference_Kind: {
|
|
|
|
const Variable& var = ((VariableReference*) lvalue)->fVariable;
|
|
|
|
if (var.fStorage == Variable::kLocal_Storage) {
|
|
|
|
(*definitions)[&var] = expr;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Expression::kSwizzle_Kind:
|
|
|
|
// We consider the variable written to as long as at least some of its components have
|
|
|
|
// been written to. This will lead to some false negatives (we won't catch it if you
|
|
|
|
// write to foo.x and then read foo.y), but being stricter could lead to false positives
|
2016-10-26 14:35:22 +00:00
|
|
|
// (we write to foo.x, and then pass foo to a function which happens to only read foo.x,
|
|
|
|
// but since we pass foo as a whole it is flagged as an error) unless we perform a much
|
2016-10-13 20:25:34 +00:00
|
|
|
// more complicated whole-program analysis. This is probably good enough.
|
2016-10-26 14:35:22 +00:00
|
|
|
this->addDefinition(((Swizzle*) lvalue)->fBase.get(),
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
2016-10-13 20:25:34 +00:00
|
|
|
definitions);
|
|
|
|
break;
|
|
|
|
case Expression::kIndex_Kind:
|
|
|
|
// see comments in Swizzle
|
2016-10-26 14:35:22 +00:00
|
|
|
this->addDefinition(((IndexExpression*) lvalue)->fBase.get(),
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
2016-10-13 20:25:34 +00:00
|
|
|
definitions);
|
|
|
|
break;
|
|
|
|
case Expression::kFieldAccess_Kind:
|
|
|
|
// see comments in Swizzle
|
2016-10-26 14:35:22 +00:00
|
|
|
this->addDefinition(((FieldAccess*) lvalue)->fBase.get(),
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
2016-10-13 20:25:34 +00:00
|
|
|
definitions);
|
|
|
|
break;
|
2018-01-18 18:32:11 +00:00
|
|
|
case Expression::kTernary_Kind:
|
|
|
|
// To simplify analysis, we just pretend that we write to both sides of the ternary.
|
|
|
|
// This allows for false positives (meaning we fail to detect that a variable might not
|
|
|
|
// have been assigned), but is preferable to false negatives.
|
|
|
|
this->addDefinition(((TernaryExpression*) lvalue)->fIfTrue.get(),
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
2018-01-18 18:32:11 +00:00
|
|
|
definitions);
|
|
|
|
this->addDefinition(((TernaryExpression*) lvalue)->fIfFalse.get(),
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
2018-01-18 18:32:11 +00:00
|
|
|
definitions);
|
|
|
|
break;
|
2016-10-13 20:25:34 +00:00
|
|
|
default:
|
|
|
|
// not an lvalue, can't happen
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(false);
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// add local variables defined by this node to the set
|
2016-10-26 14:35:22 +00:00
|
|
|
void Compiler::addDefinitions(const BasicBlock::Node& node,
|
2017-01-19 18:32:00 +00:00
|
|
|
DefinitionMap* definitions) {
|
2016-10-13 20:25:34 +00:00
|
|
|
switch (node.fKind) {
|
|
|
|
case BasicBlock::Node::kExpression_Kind: {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(node.expression());
|
2017-04-20 23:31:52 +00:00
|
|
|
const Expression* expr = (Expression*) node.expression()->get();
|
2017-01-19 18:32:00 +00:00
|
|
|
switch (expr->fKind) {
|
|
|
|
case Expression::kBinary_Kind: {
|
|
|
|
BinaryExpression* b = (BinaryExpression*) expr;
|
|
|
|
if (b->fOperator == Token::EQ) {
|
|
|
|
this->addDefinition(b->fLeft.get(), &b->fRight, definitions);
|
2017-09-11 20:50:14 +00:00
|
|
|
} else if (Compiler::IsAssignment(b->fOperator)) {
|
2017-01-19 18:32:00 +00:00
|
|
|
this->addDefinition(
|
2018-03-27 18:10:52 +00:00
|
|
|
b->fLeft.get(),
|
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
|
|
|
definitions);
|
2017-01-19 18:32:00 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2018-03-29 20:46:56 +00:00
|
|
|
case Expression::kFunctionCall_Kind: {
|
|
|
|
const FunctionCall& c = (const FunctionCall&) *expr;
|
|
|
|
for (size_t i = 0; i < c.fFunction.fParameters.size(); ++i) {
|
|
|
|
if (c.fFunction.fParameters[i]->fModifiers.fFlags & Modifiers::kOut_Flag) {
|
|
|
|
this->addDefinition(
|
|
|
|
c.fArguments[i].get(),
|
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
|
|
|
definitions);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-01-19 18:32:00 +00:00
|
|
|
case Expression::kPrefix_Kind: {
|
|
|
|
const PrefixExpression* p = (PrefixExpression*) expr;
|
|
|
|
if (p->fOperator == Token::MINUSMINUS || p->fOperator == Token::PLUSPLUS) {
|
|
|
|
this->addDefinition(
|
2018-03-27 18:10:52 +00:00
|
|
|
p->fOperand.get(),
|
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
|
|
|
definitions);
|
2017-01-19 18:32:00 +00:00
|
|
|
}
|
|
|
|
break;
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
2017-01-19 18:32:00 +00:00
|
|
|
case Expression::kPostfix_Kind: {
|
|
|
|
const PostfixExpression* p = (PostfixExpression*) expr;
|
|
|
|
if (p->fOperator == Token::MINUSMINUS || p->fOperator == Token::PLUSPLUS) {
|
|
|
|
this->addDefinition(
|
2018-03-27 18:10:52 +00:00
|
|
|
p->fOperand.get(),
|
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
|
|
|
definitions);
|
2017-01-19 18:32:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
case Expression::kVariableReference_Kind: {
|
|
|
|
const VariableReference* v = (VariableReference*) expr;
|
|
|
|
if (v->fRefKind != VariableReference::kRead_RefKind) {
|
|
|
|
this->addDefinition(
|
2018-03-27 18:10:52 +00:00
|
|
|
v,
|
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression,
|
|
|
|
definitions);
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
}
|
2017-01-19 18:32:00 +00:00
|
|
|
default:
|
|
|
|
break;
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case BasicBlock::Node::kStatement_Kind: {
|
2017-04-20 23:31:52 +00:00
|
|
|
const Statement* stmt = (Statement*) node.statement()->get();
|
2017-11-07 14:42:10 +00:00
|
|
|
if (stmt->fKind == Statement::kVarDeclaration_Kind) {
|
|
|
|
VarDeclaration& vd = (VarDeclaration&) *stmt;
|
|
|
|
if (vd.fValue) {
|
|
|
|
(*definitions)[vd.fVar] = &vd.fValue;
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::scanCFG(CFG* cfg, BlockId blockId, std::set<BlockId>* workList) {
|
|
|
|
BasicBlock& block = cfg->fBlocks[blockId];
|
|
|
|
|
|
|
|
// compute definitions after this block
|
2017-01-19 18:32:00 +00:00
|
|
|
DefinitionMap after = block.fBefore;
|
2016-10-13 20:25:34 +00:00
|
|
|
for (const BasicBlock::Node& n : block.fNodes) {
|
|
|
|
this->addDefinitions(n, &after);
|
|
|
|
}
|
|
|
|
|
|
|
|
// propagate definitions to exits
|
|
|
|
for (BlockId exitId : block.fExits) {
|
2018-03-27 18:10:52 +00:00
|
|
|
if (exitId == blockId) {
|
|
|
|
continue;
|
|
|
|
}
|
2016-10-13 20:25:34 +00:00
|
|
|
BasicBlock& exit = cfg->fBlocks[exitId];
|
|
|
|
for (const auto& pair : after) {
|
2017-01-19 18:32:00 +00:00
|
|
|
std::unique_ptr<Expression>* e1 = pair.second;
|
|
|
|
auto found = exit.fBefore.find(pair.first);
|
|
|
|
if (found == exit.fBefore.end()) {
|
|
|
|
// exit has no definition for it, just copy it
|
|
|
|
workList->insert(exitId);
|
2016-10-13 20:25:34 +00:00
|
|
|
exit.fBefore[pair.first] = e1;
|
|
|
|
} else {
|
2017-01-19 18:32:00 +00:00
|
|
|
// exit has a (possibly different) value already defined
|
|
|
|
std::unique_ptr<Expression>* e2 = exit.fBefore[pair.first];
|
2016-10-13 20:25:34 +00:00
|
|
|
if (e1 != e2) {
|
|
|
|
// definition has changed, merge and add exit block to worklist
|
|
|
|
workList->insert(exitId);
|
2017-02-27 18:26:45 +00:00
|
|
|
if (e1 && e2) {
|
|
|
|
exit.fBefore[pair.first] =
|
2018-03-27 18:10:52 +00:00
|
|
|
(std::unique_ptr<Expression>*) &fContext->fDefined_Expression;
|
2017-02-27 18:26:45 +00:00
|
|
|
} else {
|
|
|
|
exit.fBefore[pair.first] = nullptr;
|
|
|
|
}
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns a map which maps all local variables in the function to null, indicating that their value
|
|
|
|
// is initially unknown
|
2017-01-19 18:32:00 +00:00
|
|
|
static DefinitionMap compute_start_state(const CFG& cfg) {
|
|
|
|
DefinitionMap result;
|
2016-10-26 14:35:22 +00:00
|
|
|
for (const auto& block : cfg.fBlocks) {
|
|
|
|
for (const auto& node : block.fNodes) {
|
2016-10-13 20:25:34 +00:00
|
|
|
if (node.fKind == BasicBlock::Node::kStatement_Kind) {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(node.statement());
|
2017-04-20 23:31:52 +00:00
|
|
|
const Statement* s = node.statement()->get();
|
2016-10-13 20:25:34 +00:00
|
|
|
if (s->fKind == Statement::kVarDeclarations_Kind) {
|
|
|
|
const VarDeclarationsStatement* vd = (const VarDeclarationsStatement*) s;
|
2017-11-07 14:42:10 +00:00
|
|
|
for (const auto& decl : vd->fDeclaration->fVars) {
|
|
|
|
if (decl->fKind == Statement::kVarDeclaration_Kind) {
|
|
|
|
result[((VarDeclaration&) *decl).fVar] = nullptr;
|
|
|
|
}
|
2016-10-26 14:35:22 +00:00
|
|
|
}
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-04-20 23:31:52 +00:00
|
|
|
/**
|
|
|
|
* Returns true if assigning to this lvalue has no effect.
|
|
|
|
*/
|
|
|
|
static bool is_dead(const Expression& lvalue) {
|
|
|
|
switch (lvalue.fKind) {
|
|
|
|
case Expression::kVariableReference_Kind:
|
|
|
|
return ((VariableReference&) lvalue).fVariable.dead();
|
|
|
|
case Expression::kSwizzle_Kind:
|
|
|
|
return is_dead(*((Swizzle&) lvalue).fBase);
|
|
|
|
case Expression::kFieldAccess_Kind:
|
|
|
|
return is_dead(*((FieldAccess&) lvalue).fBase);
|
|
|
|
case Expression::kIndex_Kind: {
|
|
|
|
const IndexExpression& idx = (IndexExpression&) lvalue;
|
|
|
|
return is_dead(*idx.fBase) && !idx.fIndex->hasSideEffects();
|
|
|
|
}
|
2018-01-18 18:32:11 +00:00
|
|
|
case Expression::kTernary_Kind: {
|
|
|
|
const TernaryExpression& t = (TernaryExpression&) lvalue;
|
|
|
|
return !t.fTest->hasSideEffects() && is_dead(*t.fIfTrue) && is_dead(*t.fIfFalse);
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
default:
|
|
|
|
ABORT("invalid lvalue: %s\n", lvalue.description().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if this is an assignment which can be collapsed down to just the right hand side due
|
|
|
|
* to a dead target and lack of side effects on the left hand side.
|
|
|
|
*/
|
|
|
|
static bool dead_assignment(const BinaryExpression& b) {
|
2017-09-11 20:50:14 +00:00
|
|
|
if (!Compiler::IsAssignment(b.fOperator)) {
|
2017-04-20 23:31:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return is_dead(*b.fLeft);
|
|
|
|
}
|
2016-10-13 20:25:34 +00:00
|
|
|
|
2017-04-20 23:31:52 +00:00
|
|
|
void Compiler::computeDataFlow(CFG* cfg) {
|
|
|
|
cfg->fBlocks[cfg->fStart].fBefore = compute_start_state(*cfg);
|
2016-10-13 20:25:34 +00:00
|
|
|
std::set<BlockId> workList;
|
2017-04-20 23:31:52 +00:00
|
|
|
for (BlockId i = 0; i < cfg->fBlocks.size(); i++) {
|
2016-10-13 20:25:34 +00:00
|
|
|
workList.insert(i);
|
|
|
|
}
|
|
|
|
while (workList.size()) {
|
|
|
|
BlockId next = *workList.begin();
|
|
|
|
workList.erase(workList.begin());
|
2017-04-20 23:31:52 +00:00
|
|
|
this->scanCFG(cfg, next, &workList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Attempts to replace the expression pointed to by iter with a new one (in both the CFG and the
|
|
|
|
* IR). If the expression can be cleanly removed, returns true and updates the iterator to point to
|
|
|
|
* the newly-inserted element. Otherwise updates only the IR and returns false (and the CFG will
|
|
|
|
* need to be regenerated).
|
|
|
|
*/
|
|
|
|
bool try_replace_expression(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
std::unique_ptr<Expression>* newExpression) {
|
|
|
|
std::unique_ptr<Expression>* target = (*iter)->expression();
|
|
|
|
if (!b->tryRemoveExpression(iter)) {
|
|
|
|
*target = std::move(*newExpression);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
*target = std::move(*newExpression);
|
|
|
|
return b->tryInsertExpression(iter, target);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-27 20:24:51 +00:00
|
|
|
* Returns true if the expression is a constant numeric literal with the specified value, or a
|
|
|
|
* constant vector with all elements equal to the specified value.
|
2017-04-20 23:31:52 +00:00
|
|
|
*/
|
2017-04-27 20:24:51 +00:00
|
|
|
bool is_constant(const Expression& expr, double value) {
|
2017-04-20 23:31:52 +00:00
|
|
|
switch (expr.fKind) {
|
|
|
|
case Expression::kIntLiteral_Kind:
|
|
|
|
return ((IntLiteral&) expr).fValue == value;
|
|
|
|
case Expression::kFloatLiteral_Kind:
|
|
|
|
return ((FloatLiteral&) expr).fValue == value;
|
2017-04-27 20:24:51 +00:00
|
|
|
case Expression::kConstructor_Kind: {
|
|
|
|
Constructor& c = (Constructor&) expr;
|
|
|
|
if (c.fType.kind() == Type::kVector_Kind && c.isConstant()) {
|
|
|
|
for (int i = 0; i < c.fType.columns(); ++i) {
|
2019-04-19 13:39:14 +00:00
|
|
|
if (!is_constant(*c.getVecComponent(i), value)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Collapses the binary expression pointed to by iter down to just the right side (in both the IR
|
|
|
|
* and CFG structures).
|
|
|
|
*/
|
|
|
|
void delete_left(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
|
|
|
*outUpdated = true;
|
2017-05-05 14:04:06 +00:00
|
|
|
std::unique_ptr<Expression>* target = (*iter)->expression();
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*target)->fKind == Expression::kBinary_Kind);
|
2017-05-05 14:04:06 +00:00
|
|
|
BinaryExpression& bin = (BinaryExpression&) **target;
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(!bin.fLeft->hasSideEffects());
|
2017-05-05 14:04:06 +00:00
|
|
|
bool result;
|
|
|
|
if (bin.fOperator == Token::EQ) {
|
2017-05-17 14:52:55 +00:00
|
|
|
result = b->tryRemoveLValueBefore(iter, bin.fLeft.get());
|
2017-05-05 14:04:06 +00:00
|
|
|
} else {
|
2017-05-17 14:52:55 +00:00
|
|
|
result = b->tryRemoveExpressionBefore(iter, bin.fLeft.get());
|
2017-05-05 14:04:06 +00:00
|
|
|
}
|
2017-05-17 14:52:55 +00:00
|
|
|
*target = std::move(bin.fRight);
|
2017-05-05 14:04:06 +00:00
|
|
|
if (!result) {
|
2017-05-17 14:52:55 +00:00
|
|
|
*outNeedsRescan = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (*iter == b->fNodes.begin()) {
|
2017-04-20 23:31:52 +00:00
|
|
|
*outNeedsRescan = true;
|
2017-05-05 14:04:06 +00:00
|
|
|
return;
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
2017-05-05 14:04:06 +00:00
|
|
|
--(*iter);
|
2017-05-17 14:52:55 +00:00
|
|
|
if ((*iter)->fKind != BasicBlock::Node::kExpression_Kind ||
|
|
|
|
(*iter)->expression() != &bin.fRight) {
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
return;
|
|
|
|
}
|
2017-05-05 14:04:06 +00:00
|
|
|
*iter = b->fNodes.erase(*iter);
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->expression() == target);
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Collapses the binary expression pointed to by iter down to just the left side (in both the IR and
|
|
|
|
* CFG structures).
|
|
|
|
*/
|
|
|
|
void delete_right(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
|
|
|
*outUpdated = true;
|
2017-05-05 14:04:06 +00:00
|
|
|
std::unique_ptr<Expression>* target = (*iter)->expression();
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*target)->fKind == Expression::kBinary_Kind);
|
2017-05-05 14:04:06 +00:00
|
|
|
BinaryExpression& bin = (BinaryExpression&) **target;
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(!bin.fRight->hasSideEffects());
|
2017-05-05 14:04:06 +00:00
|
|
|
if (!b->tryRemoveExpressionBefore(iter, bin.fRight.get())) {
|
|
|
|
*target = std::move(bin.fLeft);
|
2017-04-20 23:31:52 +00:00
|
|
|
*outNeedsRescan = true;
|
2017-05-05 14:04:06 +00:00
|
|
|
return;
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
2017-05-17 14:52:55 +00:00
|
|
|
*target = std::move(bin.fLeft);
|
|
|
|
if (*iter == b->fNodes.begin()) {
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
return;
|
|
|
|
}
|
2017-05-05 14:04:06 +00:00
|
|
|
--(*iter);
|
2017-05-17 14:52:55 +00:00
|
|
|
if (((*iter)->fKind != BasicBlock::Node::kExpression_Kind ||
|
|
|
|
(*iter)->expression() != &bin.fLeft)) {
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
return;
|
|
|
|
}
|
2017-05-05 14:04:06 +00:00
|
|
|
*iter = b->fNodes.erase(*iter);
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->expression() == target);
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
|
2017-04-27 20:24:51 +00:00
|
|
|
/**
|
|
|
|
* Constructs the specified type using a single argument.
|
|
|
|
*/
|
|
|
|
static std::unique_ptr<Expression> construct(const Type& type, std::unique_ptr<Expression> v) {
|
|
|
|
std::vector<std::unique_ptr<Expression>> args;
|
|
|
|
args.push_back(std::move(v));
|
2017-09-11 20:50:14 +00:00
|
|
|
auto result = std::unique_ptr<Expression>(new Constructor(-1, type, std::move(args)));
|
2017-04-27 20:24:51 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used in the implementations of vectorize_left and vectorize_right. Given a vector type and an
|
|
|
|
* expression x, deletes the expression pointed to by iter and replaces it with <type>(x).
|
|
|
|
*/
|
|
|
|
static void vectorize(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
const Type& type,
|
|
|
|
std::unique_ptr<Expression>* otherExpression,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*(*iter)->expression())->fKind == Expression::kBinary_Kind);
|
|
|
|
SkASSERT(type.kind() == Type::kVector_Kind);
|
|
|
|
SkASSERT((*otherExpression)->fType.kind() == Type::kScalar_Kind);
|
2017-04-27 20:24:51 +00:00
|
|
|
*outUpdated = true;
|
|
|
|
std::unique_ptr<Expression>* target = (*iter)->expression();
|
|
|
|
if (!b->tryRemoveExpression(iter)) {
|
|
|
|
*target = construct(type, std::move(*otherExpression));
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
} else {
|
|
|
|
*target = construct(type, std::move(*otherExpression));
|
|
|
|
if (!b->tryInsertExpression(iter, target)) {
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a binary expression of the form x <op> vec<n>(y), deletes the right side and vectorizes the
|
|
|
|
* left to yield vec<n>(x).
|
|
|
|
*/
|
|
|
|
static void vectorize_left(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
|
|
|
BinaryExpression& bin = (BinaryExpression&) **(*iter)->expression();
|
|
|
|
vectorize(b, iter, bin.fRight->fType, &bin.fLeft, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a binary expression of the form vec<n>(x) <op> y, deletes the left side and vectorizes the
|
|
|
|
* right to yield vec<n>(y).
|
|
|
|
*/
|
|
|
|
static void vectorize_right(BasicBlock* b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
|
|
|
BinaryExpression& bin = (BinaryExpression&) **(*iter)->expression();
|
|
|
|
vectorize(b, iter, bin.fLeft->fType, &bin.fRight, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark that an expression which we were writing to is no longer being written to
|
|
|
|
void clear_write(const Expression& expr) {
|
|
|
|
switch (expr.fKind) {
|
|
|
|
case Expression::kVariableReference_Kind: {
|
|
|
|
((VariableReference&) expr).setRefKind(VariableReference::kRead_RefKind);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Expression::kFieldAccess_Kind:
|
|
|
|
clear_write(*((FieldAccess&) expr).fBase);
|
|
|
|
break;
|
|
|
|
case Expression::kSwizzle_Kind:
|
|
|
|
clear_write(*((Swizzle&) expr).fBase);
|
|
|
|
break;
|
|
|
|
case Expression::kIndex_Kind:
|
|
|
|
clear_write(*((IndexExpression&) expr).fBase);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ABORT("shouldn't be writing to this kind of expression\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-20 23:31:52 +00:00
|
|
|
void Compiler::simplifyExpression(DefinitionMap& definitions,
|
|
|
|
BasicBlock& b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
std::unordered_set<const Variable*>* undefinedVariables,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
|
|
|
Expression* expr = (*iter)->expression()->get();
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(expr);
|
2017-04-20 23:31:52 +00:00
|
|
|
if ((*iter)->fConstantPropagation) {
|
|
|
|
std::unique_ptr<Expression> optimized = expr->constantPropagate(*fIRGenerator, definitions);
|
|
|
|
if (optimized) {
|
2017-05-17 14:52:55 +00:00
|
|
|
*outUpdated = true;
|
2017-04-20 23:31:52 +00:00
|
|
|
if (!try_replace_expression(&b, iter, &optimized)) {
|
|
|
|
*outNeedsRescan = true;
|
2017-05-17 14:52:55 +00:00
|
|
|
return;
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->fKind == BasicBlock::Node::kExpression_Kind);
|
2017-04-20 23:31:52 +00:00
|
|
|
expr = (*iter)->expression()->get();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch (expr->fKind) {
|
|
|
|
case Expression::kVariableReference_Kind: {
|
2018-03-26 18:24:27 +00:00
|
|
|
const VariableReference& ref = (VariableReference&) *expr;
|
|
|
|
const Variable& var = ref.fVariable;
|
|
|
|
if (ref.refKind() != VariableReference::kWrite_RefKind &&
|
|
|
|
ref.refKind() != VariableReference::kPointer_RefKind &&
|
|
|
|
var.fStorage == Variable::kLocal_Storage && !definitions[&var] &&
|
2017-04-20 23:31:52 +00:00
|
|
|
(*undefinedVariables).find(&var) == (*undefinedVariables).end()) {
|
|
|
|
(*undefinedVariables).insert(&var);
|
2017-11-07 14:42:10 +00:00
|
|
|
this->error(expr->fOffset,
|
|
|
|
"'" + var.fName + "' has not been assigned");
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Expression::kTernary_Kind: {
|
|
|
|
TernaryExpression* t = (TernaryExpression*) expr;
|
|
|
|
if (t->fTest->fKind == Expression::kBoolLiteral_Kind) {
|
|
|
|
// ternary has a constant test, replace it with either the true or
|
|
|
|
// false branch
|
|
|
|
if (((BoolLiteral&) *t->fTest).fValue) {
|
|
|
|
(*iter)->setExpression(std::move(t->fIfTrue));
|
|
|
|
} else {
|
|
|
|
(*iter)->setExpression(std::move(t->fIfFalse));
|
|
|
|
}
|
|
|
|
*outUpdated = true;
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Expression::kBinary_Kind: {
|
|
|
|
BinaryExpression* bin = (BinaryExpression*) expr;
|
2017-05-05 14:04:06 +00:00
|
|
|
if (dead_assignment(*bin)) {
|
|
|
|
delete_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// collapse useless expressions like x * 1 or x + 0
|
2017-04-27 20:24:51 +00:00
|
|
|
if (((bin->fLeft->fType.kind() != Type::kScalar_Kind) &&
|
|
|
|
(bin->fLeft->fType.kind() != Type::kVector_Kind)) ||
|
|
|
|
((bin->fRight->fType.kind() != Type::kScalar_Kind) &&
|
|
|
|
(bin->fRight->fType.kind() != Type::kVector_Kind))) {
|
|
|
|
break;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
switch (bin->fOperator) {
|
|
|
|
case Token::STAR:
|
|
|
|
if (is_constant(*bin->fLeft, 1)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
if (bin->fLeft->fType.kind() == Type::kVector_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kScalar_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(1) * x -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// 1 * x -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// 1 * float4(x) -> float4(x)
|
|
|
|
// float4(1) * float4(x) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (is_constant(*bin->fLeft, 0)) {
|
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
2018-01-23 15:31:56 +00:00
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind &&
|
|
|
|
!bin->fRight->hasSideEffects()) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// 0 * float4(x) -> float4(0)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// 0 * x -> 0
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(0) * x -> float4(0)
|
|
|
|
// float4(0) * float4(x) -> float4(0)
|
2017-12-11 17:34:33 +00:00
|
|
|
if (!bin->fRight->hasSideEffects()) {
|
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
2017-04-27 20:24:51 +00:00
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
else if (is_constant(*bin->fRight, 1)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// x * float4(1) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// x * 1 -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(x) * 1 -> float4(x)
|
|
|
|
// float4(x) * float4(1) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (is_constant(*bin->fRight, 0)) {
|
|
|
|
if (bin->fLeft->fType.kind() == Type::kVector_Kind &&
|
2018-01-23 15:31:56 +00:00
|
|
|
bin->fRight->fType.kind() == Type::kScalar_Kind &&
|
|
|
|
!bin->fLeft->hasSideEffects()) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(x) * 0 -> float4(0)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// x * 0 -> 0
|
2017-07-28 19:19:46 +00:00
|
|
|
// x * float4(0) -> float4(0)
|
|
|
|
// float4(x) * float4(0) -> float4(0)
|
2017-12-11 17:34:33 +00:00
|
|
|
if (!bin->fLeft->hasSideEffects()) {
|
|
|
|
delete_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
2017-04-27 20:24:51 +00:00
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
break;
|
2017-04-21 14:23:37 +00:00
|
|
|
case Token::PLUS:
|
2017-04-20 23:31:52 +00:00
|
|
|
if (is_constant(*bin->fLeft, 0)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
if (bin->fLeft->fType.kind() == Type::kVector_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kScalar_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(0) + x -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// 0 + x -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// 0 + float4(x) -> float4(x)
|
|
|
|
// float4(0) + float4(x) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
} else if (is_constant(*bin->fRight, 0)) {
|
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// x + float4(0) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// x + 0 -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(x) + 0 -> float4(x)
|
|
|
|
// float4(x) + float4(0) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
2017-04-27 20:24:51 +00:00
|
|
|
break;
|
|
|
|
case Token::MINUS:
|
2017-04-21 14:23:37 +00:00
|
|
|
if (is_constant(*bin->fRight, 0)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// x - float4(0) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// x - 0 -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(x) - 0 -> float4(x)
|
|
|
|
// float4(x) - float4(0) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Token::SLASH:
|
|
|
|
if (is_constant(*bin->fRight, 1)) {
|
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// x / float4(1) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// x / 1 -> x
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(x) / 1 -> float4(x)
|
|
|
|
// float4(x) / float4(1) -> float4(x)
|
2017-04-27 20:24:51 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
} else if (is_constant(*bin->fLeft, 0)) {
|
|
|
|
if (bin->fLeft->fType.kind() == Type::kScalar_Kind &&
|
2018-01-23 15:31:56 +00:00
|
|
|
bin->fRight->fType.kind() == Type::kVector_Kind &&
|
|
|
|
!bin->fRight->hasSideEffects()) {
|
2017-07-28 19:19:46 +00:00
|
|
|
// 0 / float4(x) -> float4(0)
|
2017-04-27 20:24:51 +00:00
|
|
|
vectorize_left(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
} else {
|
|
|
|
// 0 / x -> 0
|
2017-07-28 19:19:46 +00:00
|
|
|
// float4(0) / x -> float4(0)
|
|
|
|
// float4(0) / float4(x) -> float4(0)
|
2017-12-11 17:34:33 +00:00
|
|
|
if (!bin->fRight->hasSideEffects()) {
|
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
2017-04-27 20:24:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Token::PLUSEQ:
|
|
|
|
if (is_constant(*bin->fRight, 0)) {
|
|
|
|
clear_write(*bin->fLeft);
|
2017-04-21 14:23:37 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
break;
|
2017-04-27 20:24:51 +00:00
|
|
|
case Token::MINUSEQ:
|
2017-04-21 14:23:37 +00:00
|
|
|
if (is_constant(*bin->fRight, 0)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
clear_write(*bin->fLeft);
|
2017-04-20 23:31:52 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
break;
|
2017-04-27 20:24:51 +00:00
|
|
|
case Token::STAREQ:
|
|
|
|
if (is_constant(*bin->fRight, 1)) {
|
|
|
|
clear_write(*bin->fLeft);
|
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Token::SLASHEQ:
|
2017-04-20 23:31:52 +00:00
|
|
|
if (is_constant(*bin->fRight, 1)) {
|
2017-04-27 20:24:51 +00:00
|
|
|
clear_write(*bin->fLeft);
|
2017-04-20 23:31:52 +00:00
|
|
|
delete_right(&b, iter, outUpdated, outNeedsRescan);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
|
2017-05-10 19:06:17 +00:00
|
|
|
// returns true if this statement could potentially execute a break at the current level (we ignore
|
|
|
|
// nested loops and switches, since any breaks inside of them will merely break the loop / switch)
|
2018-08-24 17:06:27 +00:00
|
|
|
static bool contains_conditional_break(Statement& s, bool inConditional) {
|
2017-05-10 19:06:17 +00:00
|
|
|
switch (s.fKind) {
|
|
|
|
case Statement::kBlock_Kind:
|
|
|
|
for (const auto& sub : ((Block&) s).fStatements) {
|
2018-08-24 17:06:27 +00:00
|
|
|
if (contains_conditional_break(*sub, inConditional)) {
|
2017-05-10 19:06:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
case Statement::kBreak_Kind:
|
2018-08-24 17:06:27 +00:00
|
|
|
return inConditional;
|
2017-05-10 19:06:17 +00:00
|
|
|
case Statement::kIf_Kind: {
|
|
|
|
const IfStatement& i = (IfStatement&) s;
|
2018-08-24 17:06:27 +00:00
|
|
|
return contains_conditional_break(*i.fIfTrue, true) ||
|
|
|
|
(i.fIfFalse && contains_conditional_break(*i.fIfFalse, true));
|
2017-05-10 19:06:17 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-24 17:06:27 +00:00
|
|
|
// returns true if this statement definitely executes a break at the current level (we ignore
|
|
|
|
// nested loops and switches, since any breaks inside of them will merely break the loop / switch)
|
|
|
|
static bool contains_unconditional_break(Statement& s) {
|
|
|
|
switch (s.fKind) {
|
|
|
|
case Statement::kBlock_Kind:
|
|
|
|
for (const auto& sub : ((Block&) s).fStatements) {
|
|
|
|
if (contains_unconditional_break(*sub)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
case Statement::kBreak_Kind:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-10 19:06:17 +00:00
|
|
|
// Returns a block containing all of the statements that will be run if the given case matches
|
|
|
|
// (which, owing to the statements being owned by unique_ptrs, means the switch itself will be
|
|
|
|
// broken by this call and must then be discarded).
|
|
|
|
// Returns null (and leaves the switch unmodified) if no such simple reduction is possible, such as
|
|
|
|
// when break statements appear inside conditionals.
|
|
|
|
static std::unique_ptr<Statement> block_for_case(SwitchStatement* s, SwitchCase* c) {
|
|
|
|
bool capturing = false;
|
|
|
|
std::vector<std::unique_ptr<Statement>*> statementPtrs;
|
|
|
|
for (const auto& current : s->fCases) {
|
|
|
|
if (current.get() == c) {
|
|
|
|
capturing = true;
|
|
|
|
}
|
|
|
|
if (capturing) {
|
|
|
|
for (auto& stmt : current->fStatements) {
|
2018-08-24 17:06:27 +00:00
|
|
|
if (contains_conditional_break(*stmt, s->fKind == Statement::kIf_Kind)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (contains_unconditional_break(*stmt)) {
|
2017-05-10 19:06:17 +00:00
|
|
|
capturing = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
statementPtrs.push_back(&stmt);
|
|
|
|
}
|
|
|
|
if (!capturing) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::vector<std::unique_ptr<Statement>> statements;
|
|
|
|
for (const auto& s : statementPtrs) {
|
|
|
|
statements.push_back(std::move(*s));
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
return std::unique_ptr<Statement>(new Block(-1, std::move(statements), s->fSymbols));
|
2017-05-10 19:06:17 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 23:31:52 +00:00
|
|
|
void Compiler::simplifyStatement(DefinitionMap& definitions,
|
2017-05-10 19:06:17 +00:00
|
|
|
BasicBlock& b,
|
|
|
|
std::vector<BasicBlock::Node>::iterator* iter,
|
|
|
|
std::unordered_set<const Variable*>* undefinedVariables,
|
|
|
|
bool* outUpdated,
|
|
|
|
bool* outNeedsRescan) {
|
2017-04-20 23:31:52 +00:00
|
|
|
Statement* stmt = (*iter)->statement()->get();
|
|
|
|
switch (stmt->fKind) {
|
2017-11-07 14:42:10 +00:00
|
|
|
case Statement::kVarDeclaration_Kind: {
|
|
|
|
const auto& varDecl = (VarDeclaration&) *stmt;
|
|
|
|
if (varDecl.fVar->dead() &&
|
|
|
|
(!varDecl.fValue ||
|
|
|
|
!varDecl.fValue->hasSideEffects())) {
|
|
|
|
if (varDecl.fValue) {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->statement()->get() == stmt);
|
2017-11-07 14:42:10 +00:00
|
|
|
if (!b.tryRemoveExpressionBefore(iter, varDecl.fValue.get())) {
|
|
|
|
*outNeedsRescan = true;
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
|
2017-11-07 14:42:10 +00:00
|
|
|
*outUpdated = true;
|
2017-04-20 23:31:52 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Statement::kIf_Kind: {
|
|
|
|
IfStatement& i = (IfStatement&) *stmt;
|
2017-05-10 19:06:17 +00:00
|
|
|
if (i.fTest->fKind == Expression::kBoolLiteral_Kind) {
|
|
|
|
// constant if, collapse down to a single branch
|
|
|
|
if (((BoolLiteral&) *i.fTest).fValue) {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(i.fIfTrue);
|
2017-05-10 19:06:17 +00:00
|
|
|
(*iter)->setStatement(std::move(i.fIfTrue));
|
|
|
|
} else {
|
|
|
|
if (i.fIfFalse) {
|
|
|
|
(*iter)->setStatement(std::move(i.fIfFalse));
|
|
|
|
} else {
|
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*outUpdated = true;
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
break;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
if (i.fIfFalse && i.fIfFalse->isEmpty()) {
|
|
|
|
// else block doesn't do anything, remove it
|
|
|
|
i.fIfFalse.reset();
|
|
|
|
*outUpdated = true;
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
|
|
|
if (!i.fIfFalse && i.fIfTrue->isEmpty()) {
|
|
|
|
// if block doesn't do anything, no else block
|
|
|
|
if (i.fTest->hasSideEffects()) {
|
|
|
|
// test has side effects, keep it
|
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(
|
|
|
|
new ExpressionStatement(std::move(i.fTest))));
|
|
|
|
} else {
|
|
|
|
// no if, no else, no test side effects, kill the whole if
|
|
|
|
// statement
|
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
|
|
|
|
}
|
|
|
|
*outUpdated = true;
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-05-10 19:06:17 +00:00
|
|
|
case Statement::kSwitch_Kind: {
|
|
|
|
SwitchStatement& s = (SwitchStatement&) *stmt;
|
|
|
|
if (s.fValue->isConstant()) {
|
|
|
|
// switch is constant, replace it with the case that matches
|
|
|
|
bool found = false;
|
|
|
|
SwitchCase* defaultCase = nullptr;
|
|
|
|
for (const auto& c : s.fCases) {
|
|
|
|
if (!c->fValue) {
|
|
|
|
defaultCase = c.get();
|
|
|
|
continue;
|
|
|
|
}
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(c->fValue->fKind == s.fValue->fKind);
|
2018-03-27 18:10:52 +00:00
|
|
|
found = c->fValue->compareConstant(*fContext, *s.fValue);
|
2017-05-10 19:06:17 +00:00
|
|
|
if (found) {
|
|
|
|
std::unique_ptr<Statement> newBlock = block_for_case(&s, c.get());
|
|
|
|
if (newBlock) {
|
|
|
|
(*iter)->setStatement(std::move(newBlock));
|
|
|
|
break;
|
|
|
|
} else {
|
2017-07-14 14:12:15 +00:00
|
|
|
if (s.fIsStatic && !(fFlags & kPermitInvalidStaticTests_Flag)) {
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(s.fOffset,
|
2017-05-10 19:06:17 +00:00
|
|
|
"static switch contains non-static conditional break");
|
|
|
|
s.fIsStatic = false;
|
|
|
|
}
|
|
|
|
return; // can't simplify
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
// no matching case. use default if it exists, or kill the whole thing
|
|
|
|
if (defaultCase) {
|
|
|
|
std::unique_ptr<Statement> newBlock = block_for_case(&s, defaultCase);
|
|
|
|
if (newBlock) {
|
|
|
|
(*iter)->setStatement(std::move(newBlock));
|
|
|
|
} else {
|
2017-07-14 14:12:15 +00:00
|
|
|
if (s.fIsStatic && !(fFlags & kPermitInvalidStaticTests_Flag)) {
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(s.fOffset,
|
2017-05-10 19:06:17 +00:00
|
|
|
"static switch contains non-static conditional break");
|
|
|
|
s.fIsStatic = false;
|
|
|
|
}
|
|
|
|
return; // can't simplify
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*outUpdated = true;
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
case Statement::kExpression_Kind: {
|
|
|
|
ExpressionStatement& e = (ExpressionStatement&) *stmt;
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->statement()->get() == &e);
|
2017-04-20 23:31:52 +00:00
|
|
|
if (!e.fExpression->hasSideEffects()) {
|
|
|
|
// Expression statement with no side effects, kill it
|
|
|
|
if (!b.tryRemoveExpressionBefore(iter, e.fExpression.get())) {
|
|
|
|
*outNeedsRescan = true;
|
|
|
|
}
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT((*iter)->statement()->get() == stmt);
|
2017-04-20 23:31:52 +00:00
|
|
|
(*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
|
|
|
|
*outUpdated = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::scanCFG(FunctionDefinition& f) {
|
|
|
|
CFG cfg = CFGGenerator().getCFG(f);
|
|
|
|
this->computeDataFlow(&cfg);
|
2016-10-13 20:25:34 +00:00
|
|
|
|
|
|
|
// check for unreachable code
|
|
|
|
for (size_t i = 0; i < cfg.fBlocks.size(); i++) {
|
2016-10-26 14:35:22 +00:00
|
|
|
if (i != cfg.fStart && !cfg.fBlocks[i].fEntrances.size() &&
|
2016-10-13 20:25:34 +00:00
|
|
|
cfg.fBlocks[i].fNodes.size()) {
|
2017-09-11 20:50:14 +00:00
|
|
|
int offset;
|
2017-01-19 18:32:00 +00:00
|
|
|
switch (cfg.fBlocks[i].fNodes[0].fKind) {
|
|
|
|
case BasicBlock::Node::kStatement_Kind:
|
2017-09-11 20:50:14 +00:00
|
|
|
offset = (*cfg.fBlocks[i].fNodes[0].statement())->fOffset;
|
2017-01-19 18:32:00 +00:00
|
|
|
break;
|
|
|
|
case BasicBlock::Node::kExpression_Kind:
|
2017-09-11 20:50:14 +00:00
|
|
|
offset = (*cfg.fBlocks[i].fNodes[0].expression())->fOffset;
|
2017-01-19 18:32:00 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(offset, String("unreachable"));
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (fErrorCount) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-04-20 23:31:52 +00:00
|
|
|
// check for dead code & undefined variables, perform constant propagation
|
|
|
|
std::unordered_set<const Variable*> undefinedVariables;
|
|
|
|
bool updated;
|
|
|
|
bool needsRescan = false;
|
|
|
|
do {
|
|
|
|
if (needsRescan) {
|
|
|
|
cfg = CFGGenerator().getCFG(f);
|
|
|
|
this->computeDataFlow(&cfg);
|
|
|
|
needsRescan = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
updated = false;
|
|
|
|
for (BasicBlock& b : cfg.fBlocks) {
|
|
|
|
DefinitionMap definitions = b.fBefore;
|
|
|
|
|
|
|
|
for (auto iter = b.fNodes.begin(); iter != b.fNodes.end() && !needsRescan; ++iter) {
|
|
|
|
if (iter->fKind == BasicBlock::Node::kExpression_Kind) {
|
|
|
|
this->simplifyExpression(definitions, b, &iter, &undefinedVariables, &updated,
|
|
|
|
&needsRescan);
|
|
|
|
} else {
|
|
|
|
this->simplifyStatement(definitions, b, &iter, &undefinedVariables, &updated,
|
|
|
|
&needsRescan);
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
2017-05-17 14:52:55 +00:00
|
|
|
if (needsRescan) {
|
|
|
|
break;
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
this->addDefinitions(*iter, &definitions);
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-20 23:31:52 +00:00
|
|
|
} while (updated);
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(!needsRescan);
|
2016-10-13 20:25:34 +00:00
|
|
|
|
2017-06-22 15:24:38 +00:00
|
|
|
// verify static ifs & switches, clean up dead variable decls
|
2017-05-10 19:06:17 +00:00
|
|
|
for (BasicBlock& b : cfg.fBlocks) {
|
|
|
|
DefinitionMap definitions = b.fBefore;
|
|
|
|
|
2017-06-22 15:24:38 +00:00
|
|
|
for (auto iter = b.fNodes.begin(); iter != b.fNodes.end() && !needsRescan;) {
|
2017-05-10 19:06:17 +00:00
|
|
|
if (iter->fKind == BasicBlock::Node::kStatement_Kind) {
|
|
|
|
const Statement& s = **iter->statement();
|
|
|
|
switch (s.fKind) {
|
|
|
|
case Statement::kIf_Kind:
|
2017-07-14 14:12:15 +00:00
|
|
|
if (((const IfStatement&) s).fIsStatic &&
|
|
|
|
!(fFlags & kPermitInvalidStaticTests_Flag)) {
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(s.fOffset, "static if has non-static test");
|
2017-05-10 19:06:17 +00:00
|
|
|
}
|
2017-06-22 15:24:38 +00:00
|
|
|
++iter;
|
2017-05-10 19:06:17 +00:00
|
|
|
break;
|
|
|
|
case Statement::kSwitch_Kind:
|
2017-07-14 14:12:15 +00:00
|
|
|
if (((const SwitchStatement&) s).fIsStatic &&
|
|
|
|
!(fFlags & kPermitInvalidStaticTests_Flag)) {
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(s.fOffset, "static switch has non-static test");
|
2017-05-10 19:06:17 +00:00
|
|
|
}
|
2017-06-22 15:24:38 +00:00
|
|
|
++iter;
|
|
|
|
break;
|
2017-11-07 14:42:10 +00:00
|
|
|
case Statement::kVarDeclarations_Kind: {
|
|
|
|
VarDeclarations& decls = *((VarDeclarationsStatement&) s).fDeclaration;
|
|
|
|
for (auto varIter = decls.fVars.begin(); varIter != decls.fVars.end();) {
|
|
|
|
if ((*varIter)->fKind == Statement::kNop_Kind) {
|
|
|
|
varIter = decls.fVars.erase(varIter);
|
|
|
|
} else {
|
|
|
|
++varIter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!decls.fVars.size()) {
|
|
|
|
iter = b.fNodes.erase(iter);
|
|
|
|
} else {
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-05-10 19:06:17 +00:00
|
|
|
default:
|
2017-06-22 15:24:38 +00:00
|
|
|
++iter;
|
2017-05-10 19:06:17 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-06-22 15:24:38 +00:00
|
|
|
} else {
|
|
|
|
++iter;
|
2017-05-10 19:06:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-13 20:25:34 +00:00
|
|
|
// check for missing return
|
2018-03-27 18:10:52 +00:00
|
|
|
if (f.fDeclaration.fReturnType != *fContext->fVoid_Type) {
|
2016-10-13 20:25:34 +00:00
|
|
|
if (cfg.fBlocks[cfg.fExit].fEntrances.size()) {
|
2017-09-11 20:50:14 +00:00
|
|
|
this->error(f.fOffset, String("function can exit without returning a value"));
|
2016-10-13 20:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-31 17:56:23 +00:00
|
|
|
std::unique_ptr<Program> Compiler::convertProgram(Program::Kind kind, String text,
|
2016-12-12 20:33:30 +00:00
|
|
|
const Program::Settings& settings) {
|
2016-07-01 15:22:01 +00:00
|
|
|
fErrorText = "";
|
|
|
|
fErrorCount = 0;
|
2018-04-24 17:06:09 +00:00
|
|
|
std::vector<std::unique_ptr<ProgramElement>>* inherited;
|
2016-07-25 17:08:54 +00:00
|
|
|
std::vector<std::unique_ptr<ProgramElement>> elements;
|
2016-07-01 15:22:01 +00:00
|
|
|
switch (kind) {
|
|
|
|
case Program::kVertex_Kind:
|
2018-04-24 17:06:09 +00:00
|
|
|
inherited = &fVertexInclude;
|
|
|
|
fIRGenerator->fSymbolTable = fVertexSymbolTable;
|
|
|
|
fIRGenerator->start(&settings, inherited);
|
2016-07-01 15:22:01 +00:00
|
|
|
break;
|
|
|
|
case Program::kFragment_Kind:
|
2018-04-24 17:06:09 +00:00
|
|
|
inherited = &fFragmentInclude;
|
|
|
|
fIRGenerator->fSymbolTable = fFragmentSymbolTable;
|
|
|
|
fIRGenerator->start(&settings, inherited);
|
2016-07-01 15:22:01 +00:00
|
|
|
break;
|
2017-02-16 21:37:32 +00:00
|
|
|
case Program::kGeometry_Kind:
|
2018-04-24 17:06:09 +00:00
|
|
|
inherited = &fGeometryInclude;
|
|
|
|
fIRGenerator->fSymbolTable = fGeometrySymbolTable;
|
|
|
|
fIRGenerator->start(&settings, inherited);
|
2017-02-16 21:37:32 +00:00
|
|
|
break;
|
2017-06-29 14:03:38 +00:00
|
|
|
case Program::kFragmentProcessor_Kind:
|
2018-04-24 17:06:09 +00:00
|
|
|
inherited = nullptr;
|
|
|
|
fIRGenerator->start(&settings, nullptr);
|
2018-01-24 14:52:02 +00:00
|
|
|
fIRGenerator->convertProgram(kind, SKSL_FP_INCLUDE, strlen(SKSL_FP_INCLUDE), *fTypes,
|
2017-09-18 18:10:39 +00:00
|
|
|
&elements);
|
2018-04-24 17:06:09 +00:00
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
2017-06-29 14:03:38 +00:00
|
|
|
break;
|
2019-04-23 17:31:09 +00:00
|
|
|
case Program::kPipelineStage_Kind: // fall through
|
|
|
|
case Program::kGeneric_Kind:
|
2019-04-08 13:46:01 +00:00
|
|
|
inherited = nullptr;
|
|
|
|
fIRGenerator->start(&settings, nullptr);
|
2019-04-23 17:31:09 +00:00
|
|
|
fIRGenerator->convertProgram(kind, SKSL_GENERIC_INCLUDE,
|
|
|
|
strlen(SKSL_GENERIC_INCLUDE), *fTypes, &elements);
|
2019-04-08 13:46:01 +00:00
|
|
|
fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
|
|
|
|
break;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
2017-11-10 20:34:03 +00:00
|
|
|
for (auto& element : elements) {
|
|
|
|
if (element->fKind == ProgramElement::kEnum_Kind) {
|
|
|
|
((Enum&) *element).fBuiltin = true;
|
|
|
|
}
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
std::unique_ptr<String> textPtr(new String(std::move(text)));
|
|
|
|
fSource = textPtr.get();
|
2018-01-24 14:52:02 +00:00
|
|
|
fIRGenerator->convertProgram(kind, textPtr->c_str(), textPtr->size(), *fTypes, &elements);
|
2017-09-18 18:10:39 +00:00
|
|
|
auto result = std::unique_ptr<Program>(new Program(kind,
|
|
|
|
std::move(textPtr),
|
|
|
|
settings,
|
2018-03-27 18:10:52 +00:00
|
|
|
fContext,
|
2018-04-24 17:06:09 +00:00
|
|
|
inherited,
|
2016-12-12 20:33:30 +00:00
|
|
|
std::move(elements),
|
|
|
|
fIRGenerator->fSymbolTable,
|
|
|
|
fIRGenerator->fInputs));
|
|
|
|
if (fErrorCount) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2016-07-25 17:08:54 +00:00
|
|
|
return result;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::optimize(Program& program) {
|
|
|
|
SkASSERT(!fErrorCount);
|
|
|
|
if (!program.fIsOptimized) {
|
|
|
|
program.fIsOptimized = true;
|
|
|
|
fIRGenerator->fKind = program.fKind;
|
|
|
|
fIRGenerator->fSettings = &program.fSettings;
|
|
|
|
for (auto& element : program) {
|
|
|
|
if (element.fKind == ProgramElement::kFunction_Kind) {
|
|
|
|
this->scanCFG((FunctionDefinition&) element);
|
|
|
|
}
|
|
|
|
}
|
2019-02-08 20:46:24 +00:00
|
|
|
if (program.fKind != Program::kFragmentProcessor_Kind) {
|
|
|
|
for (auto iter = program.fElements.begin(); iter != program.fElements.end();) {
|
|
|
|
if ((*iter)->fKind == ProgramElement::kVar_Kind) {
|
|
|
|
VarDeclarations& vars = (VarDeclarations&) **iter;
|
|
|
|
for (auto varIter = vars.fVars.begin(); varIter != vars.fVars.end();) {
|
|
|
|
const Variable& var = *((VarDeclaration&) **varIter).fVar;
|
|
|
|
if (var.dead()) {
|
|
|
|
varIter = vars.fVars.erase(varIter);
|
|
|
|
} else {
|
|
|
|
++varIter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (vars.fVars.size() == 0) {
|
|
|
|
iter = program.fElements.erase(iter);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
2018-07-31 13:44:36 +00:00
|
|
|
}
|
|
|
|
return fErrorCount == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<Program> Compiler::specialize(
|
|
|
|
Program& program,
|
|
|
|
const std::unordered_map<SkSL::String, SkSL::Program::Settings::Value>& inputs) {
|
|
|
|
std::vector<std::unique_ptr<ProgramElement>> elements;
|
|
|
|
for (const auto& e : program) {
|
|
|
|
elements.push_back(e.clone());
|
|
|
|
}
|
|
|
|
Program::Settings settings;
|
|
|
|
settings.fCaps = program.fSettings.fCaps;
|
|
|
|
for (auto iter = inputs.begin(); iter != inputs.end(); ++iter) {
|
|
|
|
settings.fArgs.insert(*iter);
|
|
|
|
}
|
|
|
|
std::unique_ptr<Program> result(new Program(program.fKind,
|
|
|
|
nullptr,
|
|
|
|
settings,
|
|
|
|
program.fContext,
|
|
|
|
program.fInheritedElements,
|
|
|
|
std::move(elements),
|
|
|
|
program.fSymbols,
|
|
|
|
program.fInputs));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Compiler::toSPIRV(Program& program, OutputStream& out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-16 13:56:54 +00:00
|
|
|
#ifdef SK_ENABLE_SPIRV_VALIDATION
|
2017-03-31 17:56:23 +00:00
|
|
|
StringStream buffer;
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = program.fSource.get();
|
2018-03-27 18:10:52 +00:00
|
|
|
SPIRVCodeGenerator cg(fContext.get(), &program, this, &buffer);
|
2017-03-16 13:56:54 +00:00
|
|
|
bool result = cg.generateCode();
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = nullptr;
|
2017-03-16 13:56:54 +00:00
|
|
|
if (result) {
|
|
|
|
spvtools::SpirvTools tools(SPV_ENV_VULKAN_1_0);
|
2017-06-29 14:03:38 +00:00
|
|
|
const String& data = buffer.str();
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(0 == data.size() % 4);
|
2017-03-16 13:56:54 +00:00
|
|
|
auto dumpmsg = [](spv_message_level_t, const char*, const spv_position_t&, const char* m) {
|
|
|
|
SkDebugf("SPIR-V validation error: %s\n", m);
|
|
|
|
};
|
|
|
|
tools.SetMessageConsumer(dumpmsg);
|
2018-06-12 15:05:59 +00:00
|
|
|
// Verify that the SPIR-V we produced is valid. If this SkASSERT fails, check the logs prior
|
2017-03-16 13:56:54 +00:00
|
|
|
// to the failure to see the validation errors.
|
2018-06-12 15:05:59 +00:00
|
|
|
SkAssertResult(tools.Validate((const uint32_t*) data.c_str(), data.size() / 4));
|
2017-06-29 14:03:38 +00:00
|
|
|
out.write(data.c_str(), data.size());
|
2017-03-16 13:56:54 +00:00
|
|
|
}
|
|
|
|
#else
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = program.fSource.get();
|
2018-03-27 18:10:52 +00:00
|
|
|
SPIRVCodeGenerator cg(fContext.get(), &program, this, &out);
|
2016-12-12 20:33:30 +00:00
|
|
|
bool result = cg.generateCode();
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = nullptr;
|
2017-03-16 13:56:54 +00:00
|
|
|
#endif
|
2016-07-01 15:22:01 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toSPIRV(Program& program, String* out) {
|
2017-03-31 17:56:23 +00:00
|
|
|
StringStream buffer;
|
2016-12-12 20:33:30 +00:00
|
|
|
bool result = this->toSPIRV(program, buffer);
|
|
|
|
if (result) {
|
2017-06-29 14:03:38 +00:00
|
|
|
*out = buffer.str();
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
2016-12-12 20:33:30 +00:00
|
|
|
return result;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toGLSL(Program& program, OutputStream& out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = program.fSource.get();
|
2018-03-27 18:10:52 +00:00
|
|
|
GLSLCodeGenerator cg(fContext.get(), &program, this, &out);
|
2016-12-12 20:33:30 +00:00
|
|
|
bool result = cg.generateCode();
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = nullptr;
|
2016-12-12 20:33:30 +00:00
|
|
|
return result;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toGLSL(Program& program, String* out) {
|
2017-03-31 17:56:23 +00:00
|
|
|
StringStream buffer;
|
2016-12-12 20:33:30 +00:00
|
|
|
bool result = this->toGLSL(program, buffer);
|
2016-07-01 15:22:01 +00:00
|
|
|
if (result) {
|
2017-06-29 14:03:38 +00:00
|
|
|
*out = buffer.str();
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
2016-08-03 19:43:36 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toMetal(Program& program, OutputStream& out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-03-27 18:10:52 +00:00
|
|
|
MetalCodeGenerator cg(fContext.get(), &program, this, &out);
|
2017-10-13 20:17:45 +00:00
|
|
|
bool result = cg.generateCode();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toMetal(Program& program, String* out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-07-23 20:46:16 +00:00
|
|
|
StringStream buffer;
|
|
|
|
bool result = this->toMetal(program, buffer);
|
|
|
|
if (result) {
|
|
|
|
*out = buffer.str();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toCPP(Program& program, String name, OutputStream& out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = program.fSource.get();
|
2018-03-27 18:10:52 +00:00
|
|
|
CPPCodeGenerator cg(fContext.get(), &program, this, name, &out);
|
2017-06-29 14:03:38 +00:00
|
|
|
bool result = cg.generateCode();
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = nullptr;
|
2017-06-29 14:03:38 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-31 13:44:36 +00:00
|
|
|
bool Compiler::toH(Program& program, String name, OutputStream& out) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = program.fSource.get();
|
2018-03-27 18:10:52 +00:00
|
|
|
HCodeGenerator cg(fContext.get(), &program, this, name, &out);
|
2017-06-29 14:03:38 +00:00
|
|
|
bool result = cg.generateCode();
|
2017-09-11 20:50:14 +00:00
|
|
|
fSource = nullptr;
|
2018-07-31 13:44:36 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Compiler::toPipelineStage(const Program& program, String* out,
|
|
|
|
std::vector<FormatArg>* outFormatArgs) {
|
|
|
|
SkASSERT(program.fIsOptimized);
|
|
|
|
fSource = program.fSource.get();
|
|
|
|
StringStream buffer;
|
|
|
|
PipelineStageCodeGenerator cg(fContext.get(), &program, this, &buffer, outFormatArgs);
|
|
|
|
bool result = cg.generateCode();
|
|
|
|
fSource = nullptr;
|
|
|
|
if (result) {
|
|
|
|
*out = buffer.str();
|
|
|
|
}
|
2017-06-29 14:03:38 +00:00
|
|
|
return result;
|
|
|
|
}
|
2016-12-12 20:33:30 +00:00
|
|
|
|
2019-03-21 15:05:37 +00:00
|
|
|
std::unique_ptr<ByteCode> Compiler::toByteCode(Program& program) {
|
|
|
|
if (!this->optimize(program)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
std::unique_ptr<ByteCode> result(new ByteCode());
|
|
|
|
ByteCodeGenerator cg(fContext.get(), &program, this, result.get());
|
|
|
|
if (cg.generateCode()) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2017-09-11 20:50:14 +00:00
|
|
|
const char* Compiler::OperatorName(Token::Kind kind) {
|
|
|
|
switch (kind) {
|
|
|
|
case Token::PLUS: return "+";
|
|
|
|
case Token::MINUS: return "-";
|
|
|
|
case Token::STAR: return "*";
|
|
|
|
case Token::SLASH: return "/";
|
|
|
|
case Token::PERCENT: return "%";
|
|
|
|
case Token::SHL: return "<<";
|
|
|
|
case Token::SHR: return ">>";
|
|
|
|
case Token::LOGICALNOT: return "!";
|
|
|
|
case Token::LOGICALAND: return "&&";
|
|
|
|
case Token::LOGICALOR: return "||";
|
|
|
|
case Token::LOGICALXOR: return "^^";
|
|
|
|
case Token::BITWISENOT: return "~";
|
|
|
|
case Token::BITWISEAND: return "&";
|
|
|
|
case Token::BITWISEOR: return "|";
|
|
|
|
case Token::BITWISEXOR: return "^";
|
|
|
|
case Token::EQ: return "=";
|
|
|
|
case Token::EQEQ: return "==";
|
|
|
|
case Token::NEQ: return "!=";
|
|
|
|
case Token::LT: return "<";
|
|
|
|
case Token::GT: return ">";
|
|
|
|
case Token::LTEQ: return "<=";
|
|
|
|
case Token::GTEQ: return ">=";
|
|
|
|
case Token::PLUSEQ: return "+=";
|
|
|
|
case Token::MINUSEQ: return "-=";
|
|
|
|
case Token::STAREQ: return "*=";
|
|
|
|
case Token::SLASHEQ: return "/=";
|
|
|
|
case Token::PERCENTEQ: return "%=";
|
|
|
|
case Token::SHLEQ: return "<<=";
|
|
|
|
case Token::SHREQ: return ">>=";
|
|
|
|
case Token::LOGICALANDEQ: return "&&=";
|
|
|
|
case Token::LOGICALOREQ: return "||=";
|
|
|
|
case Token::LOGICALXOREQ: return "^^=";
|
|
|
|
case Token::BITWISEANDEQ: return "&=";
|
|
|
|
case Token::BITWISEOREQ: return "|=";
|
|
|
|
case Token::BITWISEXOREQ: return "^=";
|
|
|
|
case Token::PLUSPLUS: return "++";
|
|
|
|
case Token::MINUSMINUS: return "--";
|
|
|
|
case Token::COMMA: return ",";
|
|
|
|
default:
|
|
|
|
ABORT("unsupported operator: %d\n", kind);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Compiler::IsAssignment(Token::Kind op) {
|
|
|
|
switch (op) {
|
|
|
|
case Token::EQ: // fall through
|
|
|
|
case Token::PLUSEQ: // fall through
|
|
|
|
case Token::MINUSEQ: // fall through
|
|
|
|
case Token::STAREQ: // fall through
|
|
|
|
case Token::SLASHEQ: // fall through
|
|
|
|
case Token::PERCENTEQ: // fall through
|
|
|
|
case Token::SHLEQ: // fall through
|
|
|
|
case Token::SHREQ: // fall through
|
|
|
|
case Token::BITWISEOREQ: // fall through
|
|
|
|
case Token::BITWISEXOREQ: // fall through
|
|
|
|
case Token::BITWISEANDEQ: // fall through
|
|
|
|
case Token::LOGICALOREQ: // fall through
|
|
|
|
case Token::LOGICALXOREQ: // fall through
|
|
|
|
case Token::LOGICALANDEQ:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Position Compiler::position(int offset) {
|
2018-06-12 15:05:59 +00:00
|
|
|
SkASSERT(fSource);
|
2017-09-11 20:50:14 +00:00
|
|
|
int line = 1;
|
|
|
|
int column = 1;
|
|
|
|
for (int i = 0; i < offset; i++) {
|
|
|
|
if ((*fSource)[i] == '\n') {
|
|
|
|
++line;
|
|
|
|
column = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
++column;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Position(line, column);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::error(int offset, String msg) {
|
2016-12-12 20:33:30 +00:00
|
|
|
fErrorCount++;
|
2017-09-11 20:50:14 +00:00
|
|
|
Position pos = this->position(offset);
|
|
|
|
fErrorText += "error: " + to_string(pos.fLine) + ": " + msg.c_str() + "\n";
|
2016-11-21 20:59:48 +00:00
|
|
|
}
|
|
|
|
|
2017-03-31 17:56:23 +00:00
|
|
|
String Compiler::errorText() {
|
2018-07-31 13:44:36 +00:00
|
|
|
this->writeErrorCount();
|
|
|
|
fErrorCount = 0;
|
2017-03-31 17:56:23 +00:00
|
|
|
String result = fErrorText;
|
2016-12-12 20:33:30 +00:00
|
|
|
return result;
|
2016-07-01 15:22:01 +00:00
|
|
|
}
|
|
|
|
|
2016-12-12 20:33:30 +00:00
|
|
|
void Compiler::writeErrorCount() {
|
|
|
|
if (fErrorCount) {
|
|
|
|
fErrorText += to_string(fErrorCount) + " error";
|
|
|
|
if (fErrorCount > 1) {
|
|
|
|
fErrorText += "s";
|
|
|
|
}
|
|
|
|
fErrorText += "\n";
|
2016-08-03 19:43:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-01 15:22:01 +00:00
|
|
|
} // namespace
|