/* 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 . */ /** * Optimisation stage that moves Yul variables from stack to memory. */ #pragma once #include #include #include #include #include #include namespace solidity::yul { /** * Optimisation stage that moves Yul variables from stack to memory. * It takes a map from functions names and variable names to memory offsets. * It then transforms the AST as follows: * * Single variable declarations are replaced by mstore's as follows: * If a is in the map, replace * let a * by * mstore(, 0) * respectively, replace * let a := expr * by * mstore(, expr) * * In a multi-variable declaration, variables to be moved are replaced by fresh variables and then moved to memory: * If b and d are in the map, replace * let a, b, c, d := f() * by * let _1, _2, _3, _4 := f() * mstore(, _4) * mstore(, _2) * let c := _3 * let a := _1 * * In case f has return parameters that are moved to memory, fewer variables are returned and the return values read * from memory instead. Assume the third return parameter of f (i.e. c) has to be moved to memory: * let a, b, c, d := f() * then it is replaced by * let _1, _2, _4 := f() * mstore(, _4) * mstore(, _2) * let c := mload() * let a := _1 * * Assignments to single variables are replaced by mstore's: * If a is in the map, replace * a := expr * by * mstore(, expr) * * Assignments to multiple variables are split up similarly to multi-variable declarations: * If b and d are in the map, replace * a, b, c, d := f() * by * let _1, _2, _3, _4 := f() * mstore(, _4) * mstore(, _2) * c := _3 * a := _1 * * Replace all references to a variable ``a`` in the map by ``mload()``. * * Function arguments are moved at the beginning of a function body: * If a1 is in the map, replace * function f(a1, a2, ..., a17) * { * ... * sstore(a1, a17) * } * by * function f(a1, a2, ..., a17) * { * mstore(, a1) * ... * sstore(mload(, a17) * } * This relies on the code transform popping arguments that are no longer used, if they are on the stack top. * * Functions with only one return argument that has to be moved are encapsulated in a wrapper function as follows: * Suppose b and r need to be moved in: * function f(a, b) -> r * { * ...body of f... * r := b * ...body of f continued... * } * then replace by: * function f(a, b) -> r * { * mstore(, b) * mstore(, 0) * f_1(a) * r := mload() * } * function f_1(a) * { * ...body of f... * mstore(, mload()) * ...body of f continued... * } * * Prerequisite: Disambiguator, ForLoopInitRewriter, FunctionHoister. */ class StackToMemoryMover: ASTModifier { public: /** * Runs the stack to memory mover. * @param _reservedMemory Is the amount of previously reserved memory, * i.e. the lowest memory offset to which variables can be moved. * @param _memorySlots A map from variables to a slot in memory. Based on the slot a unique offset in the memory range * between _reservedMemory and _reservedMemory + 32 * _numRequiredSlots is calculated for each * variable. * @param _numRequiredSlots The number of slots required in total. The maximum value that may occur in @a _memorySlots. */ static void run( OptimiserStepContext& _context, u256 _reservedMemory, std::map const& _memorySlots, uint64_t _numRequiredSlots, Block& _block ); using ASTModifier::operator(); void operator()(FunctionDefinition& _functionDefinition) override; void operator()(Block& _block) override; using ASTModifier::visit; void visit(Expression& _expression) override; private: class VariableMemoryOffsetTracker { public: VariableMemoryOffsetTracker( u256 _reservedMemory, std::map const& _memorySlots, uint64_t _numRequiredSlots ): m_reservedMemory(_reservedMemory), m_memorySlots(_memorySlots), m_numRequiredSlots(_numRequiredSlots) {} /// @returns a YulString containing the memory offset to be assigned to @a _variable as number literal /// or std::nullopt if the variable should not be moved. std::optional operator()(YulString _variable) const; /// @returns a YulString containing the memory offset to be assigned to @a _variable as number literal /// or std::nullopt if the variable should not be moved. std::optional operator()(TypedName const& _variable) const; /// @returns a YulString containing the memory offset to be assigned to @a _variable as number literal /// or std::nullopt if the variable should not be moved. std::optional operator()(Identifier const& _variable) const; private: u256 m_reservedMemory; std::map const& m_memorySlots; uint64_t m_numRequiredSlots = 0; }; struct FunctionMoveInfo { std::vector> returnVariableSlots; }; StackToMemoryMover( OptimiserStepContext& _context, VariableMemoryOffsetTracker const& _memoryOffsetTracker, std::map> _functionReturnVariables ); OptimiserStepContext& m_context; VariableMemoryOffsetTracker const& m_memoryOffsetTracker; NameDispenser& m_nameDispenser; /// Map from function names to the return variables of the function with that name. std::map> m_functionReturnVariables; /// List of functions generated while running this step that are to be appended to the code in the end. std::list m_newFunctionDefinitions; }; }