mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
145 lines
5.3 KiB
C++
145 lines
5.3 KiB
C++
/*
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <libyul/optimiser/StackLimitEvader.h>
|
|
#include <libyul/optimiser/CallGraphGenerator.h>
|
|
#include <libyul/optimiser/FunctionCallFinder.h>
|
|
#include <libyul/optimiser/NameDispenser.h>
|
|
#include <libyul/optimiser/StackToMemoryMover.h>
|
|
#include <libyul/backends/evm/EVMDialect.h>
|
|
#include <libyul/AsmData.h>
|
|
#include <libyul/Dialect.h>
|
|
#include <libyul/Exceptions.h>
|
|
#include <libyul/Object.h>
|
|
#include <libyul/Utilities.h>
|
|
#include <libsolutil/Algorithms.h>
|
|
#include <libsolutil/CommonData.h>
|
|
|
|
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<YulString, set<YulString>> const& unreachableVariables;
|
|
map<YulString, set<YulString>> const& callGraph;
|
|
|
|
map<YulString, uint64_t> slotAllocations{};
|
|
map<YulString, uint64_t> slotsRequiredForFunction{};
|
|
};
|
|
|
|
u256 literalArgumentValue(FunctionCall const& _call)
|
|
{
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
Literal const* literal = std::get_if<Literal>(&_call.arguments.front());
|
|
yulAssert(literal && literal->kind == LiteralKind::Number, "");
|
|
return valueOfLiteral(*literal);
|
|
}
|
|
}
|
|
|
|
void StackLimitEvader::run(
|
|
OptimiserStepContext& _context,
|
|
Object& _object,
|
|
map<YulString, set<YulString>> const& _unreachableVariables
|
|
)
|
|
{
|
|
yulAssert(_object.code, "");
|
|
auto const* evmDialect = dynamic_cast<EVMDialect const*>(&_context.dialect);
|
|
yulAssert(
|
|
evmDialect && evmDialect->providesObjectAccess(),
|
|
"StackLimitEvader can only be run on objects using the EVMDialect with object access."
|
|
);
|
|
|
|
vector<FunctionCall*> 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<uint64_t>::max() / 32, "");
|
|
reservedMemory += 32 * requiredSlots;
|
|
for (FunctionCall* memoryGuardCall: FunctionCallFinder::run(*_object.code, "memoryguard"_yulstring))
|
|
{
|
|
Literal* literal = std::get_if<Literal>(&memoryGuardCall->arguments.front());
|
|
yulAssert(literal && literal->kind == LiteralKind::Number, "");
|
|
literal->value = YulString{util::toCompactHexWithPrefix(reservedMemory)};
|
|
}
|
|
}
|