/* 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace solidity; using namespace solidity::yul; namespace { /** * Walks the call graph using a Depth-First-Search assigning memory slots to variables. * - The leaves of the call graph will get the lowest slot, increasing towards the root. * - ``slotsRequiredForFunction`` maps a function to the number of slots it requires (which is also the * next available slot that can be used by another function that calls this function). * - For each function starting from the root of the call graph: * - Visit all children that are not already visited. * - Determine the maximum value ``n`` of the values of ``slotsRequiredForFunction`` among the children. * - If the function itself contains variables that need memory slots, but is contained in a cycle, * abort the process as failure. * - If not, assign each variable its slot starting from ``n`` (incrementing it). * - Assign ``n`` to ``slotsRequiredForFunction`` of the function. */ struct MemoryOffsetAllocator { uint64_t run(YulString _function = YulString{}) { if (slotsRequiredForFunction.count(_function)) return slotsRequiredForFunction[_function]; // Assign to zero early to guard against recursive calls. slotsRequiredForFunction[_function] = 0; uint64_t requiredSlots = 0; if (callGraph.count(_function)) for (YulString child: callGraph.at(_function)) requiredSlots = std::max(run(child), requiredSlots); if (unreachableVariables.count(_function)) { yulAssert(!slotAllocations.count(_function), ""); for (YulString variable: unreachableVariables.at(_function)) if (variable.empty()) { // TODO: Too many function arguments or return parameters. } else slotAllocations[variable] = requiredSlots++; } return slotsRequiredForFunction[_function] = requiredSlots; } map> const& unreachableVariables; map> const& callGraph; map slotAllocations{}; map slotsRequiredForFunction{}; }; u256 literalArgumentValue(FunctionCall const& _call) { yulAssert(_call.arguments.size() == 1, ""); Literal const* literal = std::get_if(&_call.arguments.front()); yulAssert(literal && literal->kind == LiteralKind::Number, ""); return valueOfLiteral(*literal); } } void StackLimitEvader::run( OptimiserStepContext& _context, Object& _object, map> const& _unreachableVariables ) { yulAssert(_object.code, ""); auto const* evmDialect = dynamic_cast(&_context.dialect); yulAssert( evmDialect && evmDialect->providesObjectAccess(), "StackLimitEvader can only be run on objects using the EVMDialect with object access." ); vector memoryGuardCalls = FunctionCallFinder::run( *_object.code, "memoryguard"_yulstring ); // Do not optimise, if no ``memoryguard`` call is found. if (memoryGuardCalls.empty()) return; // Make sure all calls to ``memoryguard`` we found have the same value as argument (otherwise, abort). u256 reservedMemory = literalArgumentValue(*memoryGuardCalls.front()); for (FunctionCall const* memoryGuardCall: memoryGuardCalls) if (reservedMemory != literalArgumentValue(*memoryGuardCall)) return; CallGraph callGraph = CallGraphGenerator::callGraph(*_object.code); // We cannot move variables in recursive functions to fixed memory offsets. for (YulString function: callGraph.recursiveFunctions()) if (_unreachableVariables.count(function)) return; MemoryOffsetAllocator memoryOffsetAllocator{_unreachableVariables, callGraph.functionCalls}; uint64_t requiredSlots = memoryOffsetAllocator.run(); StackToMemoryMover::run(_context, reservedMemory, memoryOffsetAllocator.slotAllocations, requiredSlots, *_object.code); yulAssert(requiredSlots < std::numeric_limits::max() / 32, ""); reservedMemory += 32 * requiredSlots; for (FunctionCall* memoryGuardCall: FunctionCallFinder::run(*_object.code, "memoryguard"_yulstring)) { Literal* literal = std::get_if(&memoryGuardCall->arguments.front()); yulAssert(literal && literal->kind == LiteralKind::Number, ""); literal->value = YulString{util::toCompactHexWithPrefix(reservedMemory)}; } }