/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef SKSL_IRGENERATOR #define SKSL_IRGENERATOR #include "SkSLErrorReporter.h" #include "ast/SkSLASTBinaryExpression.h" #include "ast/SkSLASTBlock.h" #include "ast/SkSLASTBreakStatement.h" #include "ast/SkSLASTCallSuffix.h" #include "ast/SkSLASTContinueStatement.h" #include "ast/SkSLASTDiscardStatement.h" #include "ast/SkSLASTDoStatement.h" #include "ast/SkSLASTEnum.h" #include "ast/SkSLASTExpression.h" #include "ast/SkSLASTExpressionStatement.h" #include "ast/SkSLASTExtension.h" #include "ast/SkSLASTForStatement.h" #include "ast/SkSLASTFunction.h" #include "ast/SkSLASTIdentifier.h" #include "ast/SkSLASTIfStatement.h" #include "ast/SkSLASTInterfaceBlock.h" #include "ast/SkSLASTModifiersDeclaration.h" #include "ast/SkSLASTPrefixExpression.h" #include "ast/SkSLASTReturnStatement.h" #include "ast/SkSLASTSection.h" #include "ast/SkSLASTStatement.h" #include "ast/SkSLASTSuffixExpression.h" #include "ast/SkSLASTSwitchStatement.h" #include "ast/SkSLASTTernaryExpression.h" #include "ast/SkSLASTVarDeclaration.h" #include "ast/SkSLASTVarDeclarationStatement.h" #include "ast/SkSLASTWhileStatement.h" #include "ir/SkSLBlock.h" #include "ir/SkSLExpression.h" #include "ir/SkSLExtension.h" #include "ir/SkSLFunctionDefinition.h" #include "ir/SkSLInterfaceBlock.h" #include "ir/SkSLModifiers.h" #include "ir/SkSLModifiersDeclaration.h" #include "ir/SkSLProgram.h" #include "ir/SkSLSection.h" #include "ir/SkSLSymbolTable.h" #include "ir/SkSLStatement.h" #include "ir/SkSLType.h" #include "ir/SkSLTypeReference.h" #include "ir/SkSLVarDeclarations.h" #include "ir/SkSLVariableReference.h" namespace SkSL { /** * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding * (unoptimized) intermediate representation (IR). */ class IRGenerator { public: IRGenerator(const Context* context, std::shared_ptr root, ErrorReporter& errorReporter); void convertProgram(Program::Kind kind, const char* text, size_t length, SymbolTable& types, std::vector>* result); /** * If both operands are compile-time constants and can be folded, returns an expression * representing the folded value. Otherwise, returns null. Note that unlike most other functions * here, null does not represent a compilation error. */ std::unique_ptr constantFold(const Expression& left, Token::Kind op, const Expression& right) const; Program::Inputs fInputs; const Program::Settings* fSettings; const Context& fContext; private: /** * Prepare to compile a program. Resets state, pushes a new symbol table, and installs the * settings. */ void start(const Program::Settings* settings, std::vector>* inherited); /** * Performs cleanup after compilation is complete. */ void finish(); void pushSymbolTable(); void popSymbolTable(); std::unique_ptr convertVarDeclarations(const ASTVarDeclarations& decl, Variable::Storage storage); void convertFunction(const ASTFunction& f); std::unique_ptr convertStatement(const ASTStatement& statement); std::unique_ptr convertExpression(const ASTExpression& expression); std::unique_ptr convertModifiersDeclaration( const ASTModifiersDeclaration& m); const Type* convertType(const ASTType& type); std::unique_ptr call(int offset, const FunctionDeclaration& function, std::vector> arguments); int callCost(const FunctionDeclaration& function, const std::vector>& arguments); std::unique_ptr call(int offset, std::unique_ptr function, std::vector> arguments); int coercionCost(const Expression& expr, const Type& type); std::unique_ptr coerce(std::unique_ptr expr, const Type& type); std::unique_ptr convertAppend(int offset, const std::vector>& args); std::unique_ptr convertBlock(const ASTBlock& block); std::unique_ptr convertBreak(const ASTBreakStatement& b); std::unique_ptr convertNumberConstructor( int offset, const Type& type, std::vector> params); std::unique_ptr convertCompoundConstructor( int offset, const Type& type, std::vector> params); std::unique_ptr convertConstructor(int offset, const Type& type, std::vector> params); std::unique_ptr convertContinue(const ASTContinueStatement& c); std::unique_ptr convertDiscard(const ASTDiscardStatement& d); std::unique_ptr convertDo(const ASTDoStatement& d); std::unique_ptr convertSwitch(const ASTSwitchStatement& s); std::unique_ptr convertBinaryExpression(const ASTBinaryExpression& expression); std::unique_ptr convertExtension(const ASTExtension& e); std::unique_ptr convertExpressionStatement(const ASTExpressionStatement& s); std::unique_ptr convertFor(const ASTForStatement& f); std::unique_ptr convertIdentifier(const ASTIdentifier& identifier); std::unique_ptr convertIf(const ASTIfStatement& s); std::unique_ptr convertIndex(std::unique_ptr base, const ASTExpression& index); std::unique_ptr convertInterfaceBlock(const ASTInterfaceBlock& s); Modifiers convertModifiers(const Modifiers& m); std::unique_ptr convertPrefixExpression(const ASTPrefixExpression& expression); std::unique_ptr convertReturn(const ASTReturnStatement& r); std::unique_ptr
convertSection(const ASTSection& e); std::unique_ptr getCap(int offset, String name); std::unique_ptr getArg(int offset, String name); std::unique_ptr convertSuffixExpression(const ASTSuffixExpression& expression); std::unique_ptr convertTypeField(int offset, const Type& type, StringFragment field); std::unique_ptr convertField(std::unique_ptr base, StringFragment field); std::unique_ptr convertSwizzle(std::unique_ptr base, StringFragment fields); std::unique_ptr convertTernaryExpression(const ASTTernaryExpression& expression); std::unique_ptr convertVarDeclarationStatement(const ASTVarDeclarationStatement& s); std::unique_ptr convertWhile(const ASTWhileStatement& w); void convertEnum(const ASTEnum& e); std::unique_ptr applyInvocationIDWorkaround(std::unique_ptr main); // returns a statement which converts sk_Position from device to normalized coordinates std::unique_ptr getNormalizeSkPositionCode(); void fixRectSampling(std::vector>& arguments); void checkValid(const Expression& expr); void setRefKind(const Expression& expr, VariableReference::RefKind kind); void getConstantInt(const Expression& value, int64_t* out); Program::Kind fKind; const FunctionDeclaration* fCurrentFunction; std::unordered_map fCapsMap; std::shared_ptr fRootSymbolTable; std::shared_ptr fSymbolTable; // holds extra temp variable declarations needed for the current function std::vector> fExtraVars; int fLoopLevel; int fSwitchLevel; // count of temporary variables we have created int fTmpCount; ErrorReporter& fErrors; int fInvocations; std::vector>* fProgramElements; const Variable* fSkPerVertex = nullptr; Variable* fRTAdjust; Variable* fRTAdjustInterfaceBlock; int fRTAdjustFieldIndex; friend class AutoSymbolTable; friend class AutoLoopLevel; friend class AutoSwitchLevel; friend class Compiler; }; } #endif