/* 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 . */ // SPDX-License-Identifier: GPL-3.0 /** * Yul interpreter. */ #pragma once #include #include #include #include #include #include #include namespace solidity::yul { struct Dialect; } namespace solidity::yul::test { class InterpreterTerminatedGeneric: public util::Exception { }; class ExplicitlyTerminated: public InterpreterTerminatedGeneric { }; class ExplicitlyTerminatedWithReturn: public ExplicitlyTerminated { }; class StepLimitReached: public InterpreterTerminatedGeneric { }; class TraceLimitReached: public InterpreterTerminatedGeneric { }; class ExpressionNestingLimitReached: public InterpreterTerminatedGeneric { }; enum class ControlFlowState { Default, Continue, Break, Leave }; struct InterpreterState { bytes calldata; bytes returndata; std::map memory; /// This is different than memory.size() because we ignore gas. u256 msize; std::map storage; util::h160 address = util::h160("0x0000000000000000000000000000000011111111"); u256 balance = 0x22222222; u256 selfbalance = 0x22223333; util::h160 origin = util::h160("0x0000000000000000000000000000000033333333"); util::h160 caller = util::h160("0x0000000000000000000000000000000044444444"); u256 callvalue = 0x55555555; /// Deployed code bytes code = util::asBytes("codecodecodecodecode"); u256 gasprice = 0x66666666; util::h160 coinbase = util::h160("0x0000000000000000000000000000000077777777"); u256 timestamp = 0x88888888; u256 blockNumber = 1024; u256 difficulty = 0x9999999; u256 gaslimit = 4000000; u256 chainid = 0x01; /// The minimum value of basefee: 7 wei. u256 basefee = 0x07; /// Log of changes / effects. Sholud be structured data in the future. std::vector trace; /// This is actually an input parameter that more or less limits the runtime. size_t maxTraceSize = 0; size_t maxSteps = 0; size_t numSteps = 0; size_t maxExprNesting = 0; ControlFlowState controlFlowState = ControlFlowState::Default; /// Number of the current state instance, used for recursion protection size_t numInstance = 0; /// Prints execution trace and non-zero storage to @param _out. /// Flag @param _disableMemoryTrace, if set, does not produce a memory dump. This /// avoids false positives reports by the fuzzer when certain optimizer steps are /// activated e.g., Redundant store eliminator, Equal store eliminator. void dumpTraceAndState(std::ostream& _out, bool _disableMemoryTrace) const; /// Prints non-zero storage to @param _out. void dumpStorage(std::ostream& _out) const; bytes readMemory(u256 const& _offset, u256 const& _size) { yulAssert(_size <= 0xffff, "Too large read."); bytes data(size_t(_size), uint8_t(0)); for (size_t i = 0; i < data.size(); ++i) data[i] = memory[_offset + i]; return data; } }; /** * Scope structure built and maintained during execution. */ struct Scope { /// Used for variables and functions. Value is nullptr for variables. std::map names; std::map> subScopes; Scope* parent = nullptr; }; /** * Yul interpreter. */ class Interpreter: public ASTWalker { public: /// Executes the Yul interpreter. Flag @param _disableMemoryTracing if set ensures that /// instructions that write to memory do not affect @param _state. This /// avoids false positives reports by the fuzzer when certain optimizer steps are /// activated e.g., Redundant store eliminator, Equal store eliminator. static void run( InterpreterState& _state, Dialect const& _dialect, Block const& _ast, bool _disableExternalCalls, bool _disableMemoryTracing ); Interpreter( InterpreterState& _state, Dialect const& _dialect, Scope& _scope, bool _disableExternalCalls, bool _disableMemoryTracing, std::map _variables = {} ): m_dialect(_dialect), m_state(_state), m_variables(std::move(_variables)), m_scope(&_scope), m_disableExternalCalls(_disableExternalCalls), m_disableMemoryTrace(_disableMemoryTracing) { } void operator()(ExpressionStatement const& _statement) override; void operator()(Assignment const& _assignment) override; void operator()(VariableDeclaration const& _varDecl) override; void operator()(If const& _if) override; void operator()(Switch const& _switch) override; void operator()(FunctionDefinition const&) override; void operator()(ForLoop const&) override; void operator()(Break const&) override; void operator()(Continue const&) override; void operator()(Leave const&) override; void operator()(Block const& _block) override; bytes returnData() const { return m_state.returndata; } std::vector const& trace() const { return m_state.trace; } u256 valueOfVariable(YulString _name) const { return m_variables.at(_name); } protected: /// Asserts that the expression evaluates to exactly one value and returns it. virtual u256 evaluate(Expression const& _expression); /// Evaluates the expression and returns its value. virtual std::vector evaluateMulti(Expression const& _expression); void enterScope(Block const& _block); void leaveScope(); /// Increment interpreter step count, throwing exception if step limit /// is reached. void incrementStep(); Dialect const& m_dialect; InterpreterState& m_state; /// Values of variables. std::map m_variables; Scope* m_scope; /// If not set, external calls (e.g. using `call()`) to the same contract /// are evaluated in a new parser instance. bool m_disableExternalCalls; bool m_disableMemoryTrace; }; /** * Yul expression evaluator. */ class ExpressionEvaluator: public ASTWalker { public: ExpressionEvaluator( InterpreterState& _state, Dialect const& _dialect, Scope& _scope, std::map const& _variables, bool _disableExternalCalls, bool _disableMemoryTrace ): m_state(_state), m_dialect(_dialect), m_variables(_variables), m_scope(_scope), m_disableExternalCalls(_disableExternalCalls), m_disableMemoryTrace(_disableMemoryTrace) {} void operator()(Literal const&) override; void operator()(Identifier const&) override; void operator()(FunctionCall const& _funCall) override; /// Asserts that the expression has exactly one value and returns it. u256 value() const; /// Returns the list of values of the expression. std::vector values() const { return m_values; } protected: void runExternalCall(evmasm::Instruction _instruction); virtual std::unique_ptr makeInterpreterCopy(std::map _variables = {}) const { return std::make_unique( m_state, m_dialect, m_scope, m_disableExternalCalls, m_disableMemoryTrace, std::move(_variables) ); } virtual std::unique_ptr makeInterpreterNew(InterpreterState& _state, Scope& _scope) const { return std::make_unique( _state, m_dialect, _scope, m_disableExternalCalls, m_disableMemoryTrace ); } void setValue(u256 _value); /// Evaluates the given expression from right to left and /// stores it in m_value. void evaluateArgs( std::vector const& _expr, std::vector> const* _literalArguments ); /// Increment evaluation count, throwing exception if the /// nesting level is beyond the upper bound configured in /// the interpreter state. void incrementStep(); InterpreterState& m_state; Dialect const& m_dialect; /// Values of variables. std::map const& m_variables; Scope& m_scope; /// Current value of the expression std::vector m_values; /// Current expression nesting level unsigned m_nestingLevel = 0; bool m_disableExternalCalls; /// Flag to disable memory tracing bool m_disableMemoryTrace; }; }