/* This file is part of solidity. solidity is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. solidity is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with solidity. If not, see . */ /** * Specific AST walkers that collect semantical facts. */ #pragma once #include #include namespace yul { struct Dialect; /** * Specific AST walker that determines side-effect free-ness and movability of code. * Enters into function definitions. */ class SideEffectsCollector: public ASTWalker { public: explicit SideEffectsCollector(Dialect const& _dialect): m_dialect(_dialect) {} SideEffectsCollector(Dialect const& _dialect, Expression const& _expression); SideEffectsCollector(Dialect const& _dialect, Statement const& _statement); SideEffectsCollector(Dialect const& _dialect, Block const& _ast); using ASTWalker::operator(); void operator()(FunctionalInstruction const& _functionalInstruction) override; void operator()(FunctionCall const& _functionCall) override; bool movable() const { return m_movable; } bool sideEffectFree(bool _allowMSizeModification = false) const { if (_allowMSizeModification) return sideEffectFreeIfNoMSize(); else return m_sideEffectFree; } bool sideEffectFreeIfNoMSize() const { return m_sideEffectFreeIfNoMSize; } bool containsMSize() const { return m_containsMSize; } private: Dialect const& m_dialect; /// Is the current expression movable or not. bool m_movable = true; /// Is the current expression side-effect free, i.e. can be removed /// without changing the semantics. bool m_sideEffectFree = true; /// Is the current expression side-effect free up to msize, i.e. can be removed /// without changing the semantics except for the value returned by the msize instruction. bool m_sideEffectFreeIfNoMSize = true; /// Does the current code contain the MSize operation? /// Note that this is a purely syntactic property meaning that even if this is false, /// the code can still contain calls to functions that contain the msize instruction. bool m_containsMSize = false; }; /** * Specific AST walker that determines whether an expression is movable * and collects the referenced variables. * Can only be used on expressions. */ class MovableChecker: public SideEffectsCollector { public: explicit MovableChecker(Dialect const& _dialect): SideEffectsCollector(_dialect) {} MovableChecker(Dialect const& _dialect, Expression const& _expression); void operator()(Identifier const& _identifier) override; /// Disallow visiting anything apart from Expressions (this throws). void visit(Statement const&) override; using ASTWalker::visit; std::set const& referencedVariables() const { return m_variableReferences; } private: /// Which variables the current expression references. std::set m_variableReferences; }; /** * Helper class to find "irregular" control flow. * This includes termination, break and continue. */ class TerminationFinder { public: enum class ControlFlow { FlowOut, Break, Continue, Terminate }; TerminationFinder(Dialect const& _dialect): m_dialect(_dialect) {} /// @returns the index of the first statement in the provided sequence /// that is an unconditional ``break``, ``continue`` or a /// call to a terminating builtin function. /// If control flow can continue at the end of the list, /// returns `FlowOut` and ``size_t(-1)``. /// The function might return ``FlowOut`` even though control /// flow cannot actually continue. std::pair firstUnconditionalControlFlowChange( std::vector const& _statements ); /// @returns the control flow type of the given statement. /// This function could return FlowOut even if control flow never continues. ControlFlow controlFlowKind(Statement const& _statement); /// @returns true if the expression statement is a direct /// call to a builtin terminating function like /// ``stop``, ``revert`` or ``return``. bool isTerminatingBuiltin(ExpressionStatement const& _exprStmnt); private: Dialect const& m_dialect; }; }