mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #6767 from ethereum/constantOptimiserForYul
[Yul] Constant optimiser
This commit is contained in:
commit
c3cdae908d
@ -15,7 +15,9 @@ Compiler Features:
|
|||||||
* SMTChecker: Support tuples and function calls with multiple return values.
|
* SMTChecker: Support tuples and function calls with multiple return values.
|
||||||
* SMTChecker: Support ``delete``.
|
* SMTChecker: Support ``delete``.
|
||||||
* SMTChecker: Inline external function calls to ``this``.
|
* SMTChecker: Inline external function calls to ``this``.
|
||||||
|
* Optimizer: Add rules for multiplication and division by left-shifted one.
|
||||||
* Yul Optimizer: Simplify single-run ``for`` loops to ``if`` statements.
|
* Yul Optimizer: Simplify single-run ``for`` loops to ``if`` statements.
|
||||||
|
* Yul Optimizer: Optimize representation of numbers.
|
||||||
* Yul Optimizer: Do not inline recursive functions.
|
* Yul Optimizer: Do not inline recursive functions.
|
||||||
|
|
||||||
|
|
||||||
|
@ -91,6 +91,20 @@ inline u256 s2u(s256 _u)
|
|||||||
return u256(c_end + _u);
|
return u256(c_end + _u);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline u256 exp256(u256 _base, u256 _exponent)
|
||||||
|
{
|
||||||
|
using boost::multiprecision::limb_type;
|
||||||
|
u256 result = 1;
|
||||||
|
while (_exponent)
|
||||||
|
{
|
||||||
|
if (boost::multiprecision::bit_test(_exponent, 0))
|
||||||
|
result *= _base;
|
||||||
|
_base *= _base;
|
||||||
|
_exponent >>= 1;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
inline std::ostream& operator<<(std::ostream& os, bytes const& _bytes)
|
inline std::ostream& operator<<(std::ostream& os, bytes const& _bytes)
|
||||||
{
|
{
|
||||||
std::ostringstream ss;
|
std::ostringstream ss;
|
||||||
|
@ -33,6 +33,7 @@
|
|||||||
#include <libyul/backends/evm/AsmCodeGen.h>
|
#include <libyul/backends/evm/AsmCodeGen.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/optimiser/Suite.h>
|
#include <libyul/optimiser/Suite.h>
|
||||||
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
@ -382,7 +383,8 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
ErrorList errors;
|
ErrorList errors;
|
||||||
ErrorReporter errorReporter(errors);
|
ErrorReporter errorReporter(errors);
|
||||||
auto scanner = make_shared<langutil::Scanner>(langutil::CharStream(_assembly, "--CODEGEN--"));
|
auto scanner = make_shared<langutil::Scanner>(langutil::CharStream(_assembly, "--CODEGEN--"));
|
||||||
auto parserResult = yul::Parser(errorReporter, yul::EVMDialect::strictAssemblyForEVM(m_evmVersion)).parse(scanner, false);
|
yul::EVMDialect const& dialect = yul::EVMDialect::strictAssemblyForEVM(m_evmVersion);
|
||||||
|
auto parserResult = yul::Parser(errorReporter, dialect).parse(scanner, false);
|
||||||
#ifdef SOL_OUTPUT_ASM
|
#ifdef SOL_OUTPUT_ASM
|
||||||
cout << yul::AsmPrinter()(*parserResult) << endl;
|
cout << yul::AsmPrinter()(*parserResult) << endl;
|
||||||
#endif
|
#endif
|
||||||
@ -409,7 +411,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
analysisInfo,
|
analysisInfo,
|
||||||
errorReporter,
|
errorReporter,
|
||||||
boost::none,
|
boost::none,
|
||||||
yul::EVMDialect::strictAssemblyForEVM(m_evmVersion),
|
dialect,
|
||||||
identifierAccess.resolve
|
identifierAccess.resolve
|
||||||
).analyze(*parserResult);
|
).analyze(*parserResult);
|
||||||
if (!parserResult || !errorReporter.errors().empty() || !analyzerResult)
|
if (!parserResult || !errorReporter.errors().empty() || !analyzerResult)
|
||||||
@ -419,8 +421,10 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
// so we essentially only optimize the ABI functions.
|
// so we essentially only optimize the ABI functions.
|
||||||
if (_optimiserSettings.runYulOptimiser && _localVariables.empty())
|
if (_optimiserSettings.runYulOptimiser && _localVariables.empty())
|
||||||
{
|
{
|
||||||
|
bool const isCreation = m_runtimeContext != nullptr;
|
||||||
yul::OptimiserSuite::run(
|
yul::OptimiserSuite::run(
|
||||||
yul::EVMDialect::strictAssemblyForEVM(m_evmVersion),
|
dialect,
|
||||||
|
yul::GasMeter(dialect, isCreation, _optimiserSettings.expectedExecutionsPerDeployment),
|
||||||
*parserResult,
|
*parserResult,
|
||||||
analysisInfo,
|
analysisInfo,
|
||||||
_optimiserSettings.optimizeStackAllocation,
|
_optimiserSettings.optimizeStackAllocation,
|
||||||
@ -431,7 +435,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
analysisInfo,
|
analysisInfo,
|
||||||
errorReporter,
|
errorReporter,
|
||||||
boost::none,
|
boost::none,
|
||||||
yul::EVMDialect::strictAssemblyForEVM(m_evmVersion),
|
dialect,
|
||||||
identifierAccess.resolve
|
identifierAccess.resolve
|
||||||
).analyze(*parserResult))
|
).analyze(*parserResult))
|
||||||
reportError("Optimizer introduced error into inline assembly.");
|
reportError("Optimizer introduced error into inline assembly.");
|
||||||
|
@ -33,6 +33,7 @@
|
|||||||
#include <libyul/backends/evm/EVMObjectCompiler.h>
|
#include <libyul/backends/evm/EVMObjectCompiler.h>
|
||||||
#include <libyul/backends/wasm/WasmDialect.h>
|
#include <libyul/backends/wasm/WasmDialect.h>
|
||||||
#include <libyul/backends/wasm/EWasmObjectCompiler.h>
|
#include <libyul/backends/wasm/EWasmObjectCompiler.h>
|
||||||
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/ObjectParser.h>
|
#include <libyul/ObjectParser.h>
|
||||||
#include <libyul/optimiser/Suite.h>
|
#include <libyul/optimiser/Suite.h>
|
||||||
|
|
||||||
@ -98,7 +99,7 @@ void AssemblyStack::optimize()
|
|||||||
|
|
||||||
m_analysisSuccessful = false;
|
m_analysisSuccessful = false;
|
||||||
solAssert(m_parserResult, "");
|
solAssert(m_parserResult, "");
|
||||||
optimize(*m_parserResult);
|
optimize(*m_parserResult, true);
|
||||||
solAssert(analyzeParsed(), "Invalid source code after optimization.");
|
solAssert(analyzeParsed(), "Invalid source code after optimization.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,15 +139,18 @@ void AssemblyStack::compileEVM(AbstractAssembly& _assembly, bool _evm15, bool _o
|
|||||||
EVMObjectCompiler::compile(*m_parserResult, _assembly, *dialect, _evm15, _optimize);
|
EVMObjectCompiler::compile(*m_parserResult, _assembly, *dialect, _evm15, _optimize);
|
||||||
}
|
}
|
||||||
|
|
||||||
void AssemblyStack::optimize(Object& _object)
|
void AssemblyStack::optimize(Object& _object, bool _isCreation)
|
||||||
{
|
{
|
||||||
solAssert(_object.code, "");
|
solAssert(_object.code, "");
|
||||||
solAssert(_object.analysisInfo, "");
|
solAssert(_object.analysisInfo, "");
|
||||||
for (auto& subNode: _object.subObjects)
|
for (auto& subNode: _object.subObjects)
|
||||||
if (auto subObject = dynamic_cast<Object*>(subNode.get()))
|
if (auto subObject = dynamic_cast<Object*>(subNode.get()))
|
||||||
optimize(*subObject);
|
optimize(*subObject, false);
|
||||||
|
EVMDialect const& dialect = dynamic_cast<EVMDialect const&>(languageToDialect(m_language, m_evmVersion));
|
||||||
|
GasMeter meter(dialect, _isCreation, m_optimiserSettings.expectedExecutionsPerDeployment);
|
||||||
OptimiserSuite::run(
|
OptimiserSuite::run(
|
||||||
languageToDialect(m_language, m_evmVersion),
|
dialect,
|
||||||
|
meter,
|
||||||
*_object.code,
|
*_object.code,
|
||||||
*_object.analysisInfo,
|
*_object.analysisInfo,
|
||||||
m_optimiserSettings.optimizeStackAllocation
|
m_optimiserSettings.optimizeStackAllocation
|
||||||
|
@ -99,7 +99,7 @@ private:
|
|||||||
|
|
||||||
void compileEVM(yul::AbstractAssembly& _assembly, bool _evm15, bool _optimize) const;
|
void compileEVM(yul::AbstractAssembly& _assembly, bool _evm15, bool _optimize) const;
|
||||||
|
|
||||||
void optimize(yul::Object& _object);
|
void optimize(yul::Object& _object, bool _isCreation);
|
||||||
|
|
||||||
Language m_language = Language::Assembly;
|
Language m_language = Language::Assembly;
|
||||||
langutil::EVMVersion m_evmVersion;
|
langutil::EVMVersion m_evmVersion;
|
||||||
|
@ -58,6 +58,8 @@ add_library(yul
|
|||||||
optimiser/BlockHasher.h
|
optimiser/BlockHasher.h
|
||||||
optimiser/CommonSubexpressionEliminator.cpp
|
optimiser/CommonSubexpressionEliminator.cpp
|
||||||
optimiser/CommonSubexpressionEliminator.h
|
optimiser/CommonSubexpressionEliminator.h
|
||||||
|
optimiser/ConstantOptimiser.cpp
|
||||||
|
optimiser/ConstantOptimiser.h
|
||||||
optimiser/ControlFlowSimplifier.cpp
|
optimiser/ControlFlowSimplifier.cpp
|
||||||
optimiser/ControlFlowSimplifier.h
|
optimiser/ControlFlowSimplifier.h
|
||||||
optimiser/DataFlowAnalyzer.cpp
|
optimiser/DataFlowAnalyzer.cpp
|
||||||
|
227
libyul/optimiser/ConstantOptimiser.cpp
Normal file
227
libyul/optimiser/ConstantOptimiser.cpp
Normal file
@ -0,0 +1,227 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Optimisation stage that replaces constants by expressions that compute them.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <libyul/optimiser/ConstantOptimiser.h>
|
||||||
|
|
||||||
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
|
#include <libyul/optimiser/Metrics.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
|
#include <libyul/AsmPrinter.h>
|
||||||
|
#include <libyul/Utilities.h>
|
||||||
|
#include <libyul/AsmParser.h>
|
||||||
|
|
||||||
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
using namespace dev;
|
||||||
|
using namespace yul;
|
||||||
|
|
||||||
|
using Representation = ConstantOptimiser::Representation;
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
struct MiniEVMInterpreter: boost::static_visitor<u256>
|
||||||
|
{
|
||||||
|
explicit MiniEVMInterpreter(EVMDialect const& _dialect): m_dialect(_dialect) {}
|
||||||
|
|
||||||
|
u256 eval(Expression const& _expr)
|
||||||
|
{
|
||||||
|
return boost::apply_visitor(*this, _expr);
|
||||||
|
}
|
||||||
|
|
||||||
|
u256 eval(dev::eth::Instruction _instr, vector<Expression> const& _arguments)
|
||||||
|
{
|
||||||
|
vector<u256> args;
|
||||||
|
for (auto const& arg: _arguments)
|
||||||
|
args.emplace_back(eval(arg));
|
||||||
|
switch (_instr)
|
||||||
|
{
|
||||||
|
case eth::Instruction::ADD:
|
||||||
|
return args.at(0) + args.at(1);
|
||||||
|
case eth::Instruction::SUB:
|
||||||
|
return args.at(0) - args.at(1);
|
||||||
|
case eth::Instruction::MUL:
|
||||||
|
return args.at(0) * args.at(1);
|
||||||
|
case eth::Instruction::EXP:
|
||||||
|
return exp256(args.at(0), args.at(1));
|
||||||
|
case eth::Instruction::SHL:
|
||||||
|
return args.at(0) > 255 ? 0 : (args.at(1) << unsigned(args.at(0)));
|
||||||
|
case eth::Instruction::NOT:
|
||||||
|
return ~args.at(0);
|
||||||
|
default:
|
||||||
|
yulAssert(false, "Invalid operation generated in constant optimizer.");
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
u256 operator()(FunctionalInstruction const& _instr)
|
||||||
|
{
|
||||||
|
return eval(_instr.instruction, _instr.arguments);
|
||||||
|
}
|
||||||
|
u256 operator()(FunctionCall const& _funCall)
|
||||||
|
{
|
||||||
|
BuiltinFunctionForEVM const* fun = m_dialect.builtin(_funCall.functionName.name);
|
||||||
|
yulAssert(fun, "Expected builtin function.");
|
||||||
|
yulAssert(fun->instruction, "Expected EVM instruction.");
|
||||||
|
return eval(*fun->instruction, _funCall.arguments);
|
||||||
|
}
|
||||||
|
u256 operator()(Literal const& _literal)
|
||||||
|
{
|
||||||
|
return valueOfLiteral(_literal);
|
||||||
|
}
|
||||||
|
u256 operator()(Identifier const&) { yulAssert(false, ""); }
|
||||||
|
|
||||||
|
EVMDialect const& m_dialect;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConstantOptimiser::visit(Expression& _e)
|
||||||
|
{
|
||||||
|
if (_e.type() == typeid(Literal))
|
||||||
|
{
|
||||||
|
Literal const& literal = boost::get<Literal>(_e);
|
||||||
|
if (literal.kind != LiteralKind::Number)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (
|
||||||
|
Expression const* repr =
|
||||||
|
RepresentationFinder(m_dialect, m_meter, locationOf(_e), m_cache)
|
||||||
|
.tryFindRepresentation(valueOfLiteral(literal))
|
||||||
|
)
|
||||||
|
_e = ASTCopier{}.translate(*repr);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
ASTModifier::visit(_e);
|
||||||
|
}
|
||||||
|
|
||||||
|
Expression const* RepresentationFinder::tryFindRepresentation(dev::u256 const& _value)
|
||||||
|
{
|
||||||
|
if (_value < 0x10000)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
Representation const& repr = findRepresentation(_value);
|
||||||
|
if (repr.expression->type() == typeid(Literal))
|
||||||
|
return nullptr;
|
||||||
|
else
|
||||||
|
return repr.expression.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
Representation const& RepresentationFinder::findRepresentation(dev::u256 const& _value)
|
||||||
|
{
|
||||||
|
if (m_cache.count(_value))
|
||||||
|
return m_cache.at(_value);
|
||||||
|
|
||||||
|
Representation routine = represent(_value);
|
||||||
|
|
||||||
|
if (dev::bytesRequired(~_value) < dev::bytesRequired(_value))
|
||||||
|
// Negated is shorter to represent
|
||||||
|
routine = min(move(routine), represent("not"_yulstring, findRepresentation(~_value)));
|
||||||
|
|
||||||
|
// Decompose value into a * 2**k + b where abs(b) << 2**k
|
||||||
|
for (unsigned bits = 255; bits > 8 && m_maxSteps > 0; --bits)
|
||||||
|
{
|
||||||
|
unsigned gapDetector = unsigned((_value >> (bits - 8)) & 0x1ff);
|
||||||
|
if (gapDetector != 0xff && gapDetector != 0x100)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
u256 powerOfTwo = u256(1) << bits;
|
||||||
|
u256 upperPart = _value >> bits;
|
||||||
|
bigint lowerPart = _value & (powerOfTwo - 1);
|
||||||
|
if ((powerOfTwo - lowerPart) < lowerPart)
|
||||||
|
{
|
||||||
|
lowerPart = lowerPart - powerOfTwo; // make it negative
|
||||||
|
upperPart++;
|
||||||
|
}
|
||||||
|
if (upperPart == 0)
|
||||||
|
continue;
|
||||||
|
if (abs(lowerPart) >= (powerOfTwo >> 8))
|
||||||
|
continue;
|
||||||
|
Representation newRoutine;
|
||||||
|
if (m_dialect.evmVersion().hasBitwiseShifting())
|
||||||
|
newRoutine = represent("shl"_yulstring, represent(bits), findRepresentation(upperPart));
|
||||||
|
else
|
||||||
|
{
|
||||||
|
newRoutine = represent("exp"_yulstring, represent(2), represent(bits));
|
||||||
|
if (upperPart != 1)
|
||||||
|
newRoutine = represent("mul"_yulstring, findRepresentation(upperPart), newRoutine);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newRoutine.cost >= routine.cost)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (lowerPart > 0)
|
||||||
|
newRoutine = represent("add"_yulstring, newRoutine, findRepresentation(u256(abs(lowerPart))));
|
||||||
|
else if (lowerPart < 0)
|
||||||
|
newRoutine = represent("sub"_yulstring, newRoutine, findRepresentation(u256(abs(lowerPart))));
|
||||||
|
|
||||||
|
if (m_maxSteps > 0)
|
||||||
|
m_maxSteps--;
|
||||||
|
routine = min(move(routine), move(newRoutine));
|
||||||
|
}
|
||||||
|
yulAssert(MiniEVMInterpreter{m_dialect}.eval(*routine.expression) == _value, "Invalid expression generated.");
|
||||||
|
return m_cache[_value] = move(routine);
|
||||||
|
}
|
||||||
|
|
||||||
|
Representation RepresentationFinder::represent(dev::u256 const& _value) const
|
||||||
|
{
|
||||||
|
Representation repr;
|
||||||
|
repr.expression = make_unique<Expression>(Literal{m_location, LiteralKind::Number, YulString{formatNumber(_value)}, {}});
|
||||||
|
repr.cost = m_meter.costs(*repr.expression);
|
||||||
|
return repr;
|
||||||
|
}
|
||||||
|
|
||||||
|
Representation RepresentationFinder::represent(
|
||||||
|
YulString _instruction,
|
||||||
|
Representation const& _argument
|
||||||
|
) const
|
||||||
|
{
|
||||||
|
Representation repr;
|
||||||
|
repr.expression = make_unique<Expression>(FunctionCall{
|
||||||
|
m_location,
|
||||||
|
Identifier{m_location, _instruction},
|
||||||
|
{ASTCopier{}.translate(*_argument.expression)}
|
||||||
|
});
|
||||||
|
repr.cost = _argument.cost + m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction);
|
||||||
|
return repr;
|
||||||
|
}
|
||||||
|
|
||||||
|
Representation RepresentationFinder::represent(
|
||||||
|
YulString _instruction,
|
||||||
|
Representation const& _arg1,
|
||||||
|
Representation const& _arg2
|
||||||
|
) const
|
||||||
|
{
|
||||||
|
Representation repr;
|
||||||
|
repr.expression = make_unique<Expression>(FunctionCall{
|
||||||
|
m_location,
|
||||||
|
Identifier{m_location, _instruction},
|
||||||
|
{ASTCopier{}.translate(*_arg1.expression), ASTCopier{}.translate(*_arg2.expression)}
|
||||||
|
});
|
||||||
|
repr.cost = m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction) + _arg1.cost + _arg2.cost;
|
||||||
|
return repr;
|
||||||
|
}
|
||||||
|
|
||||||
|
Representation RepresentationFinder::min(Representation _a, Representation _b)
|
||||||
|
{
|
||||||
|
if (_a.cost <= _b.cost)
|
||||||
|
return _a;
|
||||||
|
else
|
||||||
|
return _b;
|
||||||
|
}
|
108
libyul/optimiser/ConstantOptimiser.h
Normal file
108
libyul/optimiser/ConstantOptimiser.h
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Optimisation stage that replaces constants by expressions that compute them.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
#include <libyul/YulString.h>
|
||||||
|
#include <libyul/Dialect.h>
|
||||||
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
|
#include <liblangutil/SourceLocation.h>
|
||||||
|
|
||||||
|
#include <libdevcore/Common.h>
|
||||||
|
|
||||||
|
#include <tuple>
|
||||||
|
#include <map>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace yul
|
||||||
|
{
|
||||||
|
struct Dialect;
|
||||||
|
class GasMeter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optimisation stage that replaces constants by expressions that compute them.
|
||||||
|
*
|
||||||
|
* Prerequisite: None
|
||||||
|
*/
|
||||||
|
class ConstantOptimiser: public ASTModifier
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ConstantOptimiser(EVMDialect const& _dialect, GasMeter const& _meter):
|
||||||
|
m_dialect(_dialect),
|
||||||
|
m_meter(_meter)
|
||||||
|
{}
|
||||||
|
|
||||||
|
void visit(Expression& _e) override;
|
||||||
|
|
||||||
|
struct Representation
|
||||||
|
{
|
||||||
|
std::unique_ptr<Expression> expression;
|
||||||
|
size_t cost = size_t(-1);
|
||||||
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
|
EVMDialect const& m_dialect;
|
||||||
|
GasMeter const& m_meter;
|
||||||
|
std::map<dev::u256, Representation> m_cache;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RepresentationFinder
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
using Representation = ConstantOptimiser::Representation;
|
||||||
|
RepresentationFinder(
|
||||||
|
EVMDialect const& _dialect,
|
||||||
|
GasMeter const& _meter,
|
||||||
|
langutil::SourceLocation _location,
|
||||||
|
std::map<dev::u256, Representation>& _cache
|
||||||
|
):
|
||||||
|
m_dialect(_dialect),
|
||||||
|
m_meter(_meter),
|
||||||
|
m_location(std::move(_location)),
|
||||||
|
m_cache(_cache)
|
||||||
|
{}
|
||||||
|
|
||||||
|
/// @returns a cheaper representation for the number than its representation
|
||||||
|
/// as a literal or nullptr otherwise.
|
||||||
|
Expression const* tryFindRepresentation(dev::u256 const& _value);
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// Recursively try to find the cheapest representation of the given number,
|
||||||
|
/// literal if necessary.
|
||||||
|
Representation const& findRepresentation(dev::u256 const& _value);
|
||||||
|
|
||||||
|
Representation represent(dev::u256 const& _value) const;
|
||||||
|
Representation represent(YulString _instruction, Representation const& _arg) const;
|
||||||
|
Representation represent(YulString _instruction, Representation const& _arg1, Representation const& _arg2) const;
|
||||||
|
|
||||||
|
Representation min(Representation _a, Representation _b);
|
||||||
|
|
||||||
|
EVMDialect const& m_dialect;
|
||||||
|
GasMeter const& m_meter;
|
||||||
|
langutil::SourceLocation m_location;
|
||||||
|
/// Counter for the complexity of optimization, will stop when it reaches zero.
|
||||||
|
size_t m_maxSteps = 10000;
|
||||||
|
std::map<dev::u256, Representation>& m_cache;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
@ -22,11 +22,14 @@
|
|||||||
|
|
||||||
#include <libyul/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/Utilities.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
|
|
||||||
#include <libevmasm/Instruction.h>
|
#include <libevmasm/Instruction.h>
|
||||||
|
#include <libevmasm/GasMeter.h>
|
||||||
|
|
||||||
#include <libdevcore/Visitor.h>
|
#include <libdevcore/Visitor.h>
|
||||||
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
@ -166,6 +169,93 @@ void CodeCost::addInstructionCost(eth::Instruction _instruction)
|
|||||||
m_cost += 49;
|
m_cost += 49;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t GasMeter::costs(Expression const& _expression) const
|
||||||
|
{
|
||||||
|
return combineCosts(GasMeterVisitor::costs(_expression, m_dialect, m_isCreation));
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GasMeter::instructionCosts(eth::Instruction _instruction) const
|
||||||
|
{
|
||||||
|
return combineCosts(GasMeterVisitor::instructionCosts(_instruction, m_dialect, m_isCreation));
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GasMeter::combineCosts(std::pair<size_t, size_t> _costs) const
|
||||||
|
{
|
||||||
|
return _costs.first * m_runs + _costs.second;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
pair<size_t, size_t> GasMeterVisitor::costs(
|
||||||
|
Expression const& _expression,
|
||||||
|
EVMDialect const& _dialect,
|
||||||
|
bool _isCreation
|
||||||
|
)
|
||||||
|
{
|
||||||
|
GasMeterVisitor gmv(_dialect, _isCreation);
|
||||||
|
gmv.visit(_expression);
|
||||||
|
return {gmv.m_runGas, gmv.m_dataGas};
|
||||||
|
}
|
||||||
|
|
||||||
|
pair<size_t, size_t> GasMeterVisitor::instructionCosts(
|
||||||
|
dev::eth::Instruction _instruction,
|
||||||
|
EVMDialect const& _dialect,
|
||||||
|
bool _isCreation
|
||||||
|
)
|
||||||
|
{
|
||||||
|
GasMeterVisitor gmv(_dialect, _isCreation);
|
||||||
|
gmv.instructionCostsInternal(_instruction);
|
||||||
|
return {gmv.m_runGas, gmv.m_dataGas};
|
||||||
|
}
|
||||||
|
|
||||||
|
void GasMeterVisitor::operator()(FunctionCall const& _funCall)
|
||||||
|
{
|
||||||
|
ASTWalker::operator()(_funCall);
|
||||||
|
if (BuiltinFunctionForEVM const* f = m_dialect.builtin(_funCall.functionName.name))
|
||||||
|
if (f->instruction)
|
||||||
|
{
|
||||||
|
instructionCostsInternal(*f->instruction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
yulAssert(false, "Functions not implemented.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void GasMeterVisitor::operator()(FunctionalInstruction const& _fun)
|
||||||
|
{
|
||||||
|
ASTWalker::operator()(_fun);
|
||||||
|
instructionCostsInternal(_fun.instruction);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GasMeterVisitor::operator()(Literal const& _lit)
|
||||||
|
{
|
||||||
|
m_runGas += dev::eth::GasMeter::runGas(dev::eth::Instruction::PUSH1);
|
||||||
|
m_dataGas +=
|
||||||
|
singleByteDataGas() +
|
||||||
|
size_t(dev::eth::GasMeter::dataGas(dev::toCompactBigEndian(valueOfLiteral(_lit), 1), m_isCreation));
|
||||||
|
}
|
||||||
|
|
||||||
|
void GasMeterVisitor::operator()(Identifier const&)
|
||||||
|
{
|
||||||
|
m_runGas += dev::eth::GasMeter::runGas(dev::eth::Instruction::DUP1);
|
||||||
|
m_dataGas += singleByteDataGas();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GasMeterVisitor::singleByteDataGas() const
|
||||||
|
{
|
||||||
|
if (m_isCreation)
|
||||||
|
return dev::eth::GasCosts::txDataNonZeroGas;
|
||||||
|
else
|
||||||
|
return dev::eth::GasCosts::createDataGas;
|
||||||
|
}
|
||||||
|
|
||||||
|
void GasMeterVisitor::instructionCostsInternal(dev::eth::Instruction _instruction)
|
||||||
|
{
|
||||||
|
if (_instruction == eth::Instruction::EXP)
|
||||||
|
m_runGas += dev::eth::GasCosts::expGas + dev::eth::GasCosts::expByteGas(m_dialect.evmVersion());
|
||||||
|
else
|
||||||
|
m_runGas += dev::eth::GasMeter::runGas(_instruction);
|
||||||
|
m_dataGas += singleByteDataGas();
|
||||||
|
}
|
||||||
|
|
||||||
void AssignmentCounter::operator()(Assignment const& _assignment)
|
void AssignmentCounter::operator()(Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
for (auto const& variable: _assignment.variableNames)
|
for (auto const& variable: _assignment.variableNames)
|
||||||
|
@ -21,6 +21,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
#include <liblangutil/EVMVersion.h>
|
||||||
|
#include <libevmasm/Instruction.h>
|
||||||
|
|
||||||
#include <libevmasm/Instruction.h>
|
#include <libevmasm/Instruction.h>
|
||||||
|
|
||||||
@ -28,6 +30,7 @@ namespace yul
|
|||||||
{
|
{
|
||||||
|
|
||||||
struct Dialect;
|
struct Dialect;
|
||||||
|
struct EVMDialect;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Metric for the size of code.
|
* Metric for the size of code.
|
||||||
@ -93,6 +96,75 @@ private:
|
|||||||
size_t m_cost = 0;
|
size_t m_cost = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gas meter for expressions only involving literals, identifiers and
|
||||||
|
* EVM instructions.
|
||||||
|
*
|
||||||
|
* Assumes that EXP is not used with exponents larger than a single byte.
|
||||||
|
* Is not particularly exact for anything apart from arithmetic.
|
||||||
|
*/
|
||||||
|
class GasMeter
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
GasMeter(EVMDialect const& _dialect, bool _isCreation, size_t _runs):
|
||||||
|
m_dialect(_dialect),
|
||||||
|
m_isCreation{_isCreation},
|
||||||
|
m_runs(_runs)
|
||||||
|
{}
|
||||||
|
|
||||||
|
/// @returns the full combined costs of deploying and evaluating the expression.
|
||||||
|
size_t costs(Expression const& _expression) const;
|
||||||
|
/// @returns the combined costs of deploying and running the instruction, not including
|
||||||
|
/// the costs for its arguments.
|
||||||
|
size_t instructionCosts(dev::eth::Instruction _instruction) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
size_t combineCosts(std::pair<size_t, size_t> _costs) const;
|
||||||
|
|
||||||
|
EVMDialect const& m_dialect;
|
||||||
|
bool m_isCreation = false;
|
||||||
|
size_t m_runs;
|
||||||
|
};
|
||||||
|
|
||||||
|
class GasMeterVisitor: public ASTWalker
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static std::pair<size_t, size_t> costs(
|
||||||
|
Expression const& _expression,
|
||||||
|
EVMDialect const& _dialect,
|
||||||
|
bool _isCreation
|
||||||
|
);
|
||||||
|
|
||||||
|
static std::pair<size_t, size_t> instructionCosts(
|
||||||
|
dev::eth::Instruction _instruction,
|
||||||
|
EVMDialect const& _dialect,
|
||||||
|
bool _isCreation = false
|
||||||
|
);
|
||||||
|
|
||||||
|
public:
|
||||||
|
GasMeterVisitor(EVMDialect const& _dialect, bool _isCreation):
|
||||||
|
m_dialect(_dialect),
|
||||||
|
m_isCreation{_isCreation}
|
||||||
|
{}
|
||||||
|
|
||||||
|
void operator()(FunctionCall const& _funCall) override;
|
||||||
|
void operator()(FunctionalInstruction const& _instr) override;
|
||||||
|
void operator()(Literal const& _literal) override;
|
||||||
|
void operator()(Identifier const& _identifier) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
size_t singleByteDataGas() const;
|
||||||
|
/// Computes the cost of storing and executing the single instruction (excluding its arguments).
|
||||||
|
/// For EXP, it assumes that the exponent is at most 255.
|
||||||
|
/// Does not work particularly exact for anything apart from arithmetic.
|
||||||
|
void instructionCostsInternal(dev::eth::Instruction _instruction);
|
||||||
|
|
||||||
|
EVMDialect const& m_dialect;
|
||||||
|
bool m_isCreation = false;
|
||||||
|
size_t m_runGas = 0;
|
||||||
|
size_t m_dataGas = 0;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Counts the number of assignments to every variable.
|
* Counts the number of assignments to every variable.
|
||||||
* Only works after running the Disambiguator.
|
* Only works after running the Disambiguator.
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
#include <libyul/optimiser/VarDeclInitializer.h>
|
#include <libyul/optimiser/VarDeclInitializer.h>
|
||||||
#include <libyul/optimiser/BlockFlattener.h>
|
#include <libyul/optimiser/BlockFlattener.h>
|
||||||
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
||||||
|
#include <libyul/optimiser/ConstantOptimiser.h>
|
||||||
#include <libyul/optimiser/DeadCodeEliminator.h>
|
#include <libyul/optimiser/DeadCodeEliminator.h>
|
||||||
#include <libyul/optimiser/FunctionGrouper.h>
|
#include <libyul/optimiser/FunctionGrouper.h>
|
||||||
#include <libyul/optimiser/FunctionHoister.h>
|
#include <libyul/optimiser/FunctionHoister.h>
|
||||||
@ -59,6 +60,7 @@ using namespace yul;
|
|||||||
|
|
||||||
void OptimiserSuite::run(
|
void OptimiserSuite::run(
|
||||||
Dialect const& _dialect,
|
Dialect const& _dialect,
|
||||||
|
GasMeter const& _meter,
|
||||||
Block& _ast,
|
Block& _ast,
|
||||||
AsmAnalysisInfo const& _analysisInfo,
|
AsmAnalysisInfo const& _analysisInfo,
|
||||||
bool _optimizeStackAllocation,
|
bool _optimizeStackAllocation,
|
||||||
@ -206,6 +208,8 @@ void OptimiserSuite::run(
|
|||||||
ControlFlowSimplifier{_dialect}(ast);
|
ControlFlowSimplifier{_dialect}(ast);
|
||||||
|
|
||||||
FunctionGrouper{}(ast);
|
FunctionGrouper{}(ast);
|
||||||
|
|
||||||
|
ConstantOptimiser{dynamic_cast<EVMDialect const&>(_dialect), _meter}(ast);
|
||||||
VarNameCleaner{ast, _dialect, reservedIdentifiers}(ast);
|
VarNameCleaner{ast, _dialect, reservedIdentifiers}(ast);
|
||||||
yul::AsmAnalyzer::analyzeStrictAssertCorrect(_dialect, ast);
|
yul::AsmAnalyzer::analyzeStrictAssertCorrect(_dialect, ast);
|
||||||
|
|
||||||
|
@ -31,6 +31,7 @@ namespace yul
|
|||||||
|
|
||||||
struct AsmAnalysisInfo;
|
struct AsmAnalysisInfo;
|
||||||
struct Dialect;
|
struct Dialect;
|
||||||
|
class GasMeter;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Optimiser suite that combines all steps and also provides the settings for the heuristics
|
* Optimiser suite that combines all steps and also provides the settings for the heuristics
|
||||||
@ -40,6 +41,7 @@ class OptimiserSuite
|
|||||||
public:
|
public:
|
||||||
static void run(
|
static void run(
|
||||||
Dialect const& _dialect,
|
Dialect const& _dialect,
|
||||||
|
GasMeter const& _meter,
|
||||||
Block& _ast,
|
Block& _ast,
|
||||||
AsmAnalysisInfo const& _analysisInfo,
|
AsmAnalysisInfo const& _analysisInfo,
|
||||||
bool _optimizeStackAllocation,
|
bool _optimizeStackAllocation,
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include <libyul/optimiser/VarDeclInitializer.h>
|
#include <libyul/optimiser/VarDeclInitializer.h>
|
||||||
#include <libyul/optimiser/VarNameCleaner.h>
|
#include <libyul/optimiser/VarNameCleaner.h>
|
||||||
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
#include <libyul/optimiser/ControlFlowSimplifier.h>
|
||||||
|
#include <libyul/optimiser/ConstantOptimiser.h>
|
||||||
#include <libyul/optimiser/DeadCodeEliminator.h>
|
#include <libyul/optimiser/DeadCodeEliminator.h>
|
||||||
#include <libyul/optimiser/Disambiguator.h>
|
#include <libyul/optimiser/Disambiguator.h>
|
||||||
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
|
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
|
||||||
@ -46,6 +47,7 @@
|
|||||||
#include <libyul/optimiser/StructuralSimplifier.h>
|
#include <libyul/optimiser/StructuralSimplifier.h>
|
||||||
#include <libyul/optimiser/StackCompressor.h>
|
#include <libyul/optimiser/StackCompressor.h>
|
||||||
#include <libyul/optimiser/Suite.h>
|
#include <libyul/optimiser/Suite.h>
|
||||||
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/backends/wasm/WordSizeTransform.h>
|
#include <libyul/backends/wasm/WordSizeTransform.h>
|
||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
@ -118,6 +120,11 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
|||||||
disambiguate();
|
disambiguate();
|
||||||
BlockFlattener{}(*m_ast);
|
BlockFlattener{}(*m_ast);
|
||||||
}
|
}
|
||||||
|
else if (m_optimizerStep == "constantOptimiser")
|
||||||
|
{
|
||||||
|
GasMeter meter(dynamic_cast<EVMDialect const&>(*m_dialect), false, 200);
|
||||||
|
ConstantOptimiser{dynamic_cast<EVMDialect const&>(*m_dialect), meter}(*m_ast);
|
||||||
|
}
|
||||||
else if (m_optimizerStep == "varDeclInitializer")
|
else if (m_optimizerStep == "varDeclInitializer")
|
||||||
VarDeclInitializer{}(*m_ast);
|
VarDeclInitializer{}(*m_ast);
|
||||||
else if (m_optimizerStep == "varNameCleaner")
|
else if (m_optimizerStep == "varNameCleaner")
|
||||||
@ -288,7 +295,10 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
|||||||
WordSizeTransform::run(*m_ast, nameDispenser);
|
WordSizeTransform::run(*m_ast, nameDispenser);
|
||||||
}
|
}
|
||||||
else if (m_optimizerStep == "fullSuite")
|
else if (m_optimizerStep == "fullSuite")
|
||||||
OptimiserSuite::run(*m_dialect, *m_ast, *m_analysisInfo, true);
|
{
|
||||||
|
GasMeter meter(dynamic_cast<EVMDialect const&>(*m_dialect), false, 200);
|
||||||
|
OptimiserSuite::run(*m_dialect, meter, *m_ast, *m_analysisInfo, true);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
AnsiColorized(_stream, _formatted, {formatting::BOLD, formatting::RED}) << _linePrefix << "Invalid optimizer step: " << m_optimizerStep << endl;
|
AnsiColorized(_stream, _formatted, {formatting::BOLD, formatting::RED}) << _linePrefix << "Invalid optimizer step: " << m_optimizerStep << endl;
|
||||||
|
@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
let x := 0xff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00
|
||||||
|
let y := 0x1100ff00ff00ff001100ff00ff001100ff00ff001100ff00ff001100ff001100
|
||||||
|
let z := 0xffff0000ffff0000ffff0000ffff0000ff00ff00ffff0000ffff0000ffff0000
|
||||||
|
let w := 0xffffffff000000ffffef000001feff000067ffefff0000ff230002ffee00fff7
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: constantOptimiser
|
||||||
|
// ----
|
||||||
|
// {
|
||||||
|
// let x := 0xff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00ff00
|
||||||
|
// let y := 0x1100ff00ff00ff001100ff00ff001100ff00ff001100ff00ff001100ff001100
|
||||||
|
// let z := 0xffff0000ffff0000ffff0000ffff0000ff00ff00ffff0000ffff0000ffff0000
|
||||||
|
// let w := 0xffffffff000000ffffef000001feff000067ffefff0000ff230002ffee00fff7
|
||||||
|
// }
|
20
test/libyul/yulOptimizerTests/constantOptimiser/gaps.yul
Normal file
20
test/libyul/yulOptimizerTests/constantOptimiser/gaps.yul
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
let a := 0x10000000000000000000000000000000000000000000
|
||||||
|
let x := 0x11000000000000000000000000000000000000ffffffffffffffffffffffff23
|
||||||
|
let y := 0xfffffffff00000000000000000000000000000000000000000000000000000ff
|
||||||
|
for { let i := 0xff00000000000 } lt(i, 873687623427364) { i := add(i, 3234234234234) } {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// EVMVersion: >=constantinople
|
||||||
|
// step: constantOptimiser
|
||||||
|
// ----
|
||||||
|
// {
|
||||||
|
// let a := shl(172, 1)
|
||||||
|
// let x := add(shl(248, 17), 0xffffffffffffffffffffffff23)
|
||||||
|
// let y := add(shl(220, 0x0fffffffff), 255)
|
||||||
|
// for { let i := 0xff00000000000 }
|
||||||
|
// lt(i, 873687623427364)
|
||||||
|
// { i := add(i, 3234234234234) }
|
||||||
|
// { }
|
||||||
|
// }
|
@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
let x := 8
|
||||||
|
let y := 0xffff
|
||||||
|
for { let i := 0xff00 } lt(i, 2) { i := add(i, 3) } {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// step: constantOptimiser
|
||||||
|
// ----
|
||||||
|
// {
|
||||||
|
// let x := 8
|
||||||
|
// let y := 0xffff
|
||||||
|
// for { let i := 0xff00 } lt(i, 2) { i := add(i, 3) }
|
||||||
|
// { }
|
||||||
|
// }
|
@ -1,3 +1,4 @@
|
|||||||
|
// We require constantinople because numbers are represented using shifts.
|
||||||
{
|
{
|
||||||
// This ignores many of the encoding / decoding functions. Over time,
|
// This ignores many of the encoding / decoding functions. Over time,
|
||||||
// we should add them all here.
|
// we should add them all here.
|
||||||
@ -1071,13 +1072,14 @@
|
|||||||
}
|
}
|
||||||
// ====
|
// ====
|
||||||
// step: fullSuite
|
// step: fullSuite
|
||||||
|
// EVMVersion: >=constantinople
|
||||||
// ----
|
// ----
|
||||||
// {
|
// {
|
||||||
// {
|
// {
|
||||||
// let _1 := mload(1)
|
// let _1 := mload(1)
|
||||||
// let _2 := mload(0)
|
// let _2 := mload(0)
|
||||||
// if slt(sub(_1, _2), 64) { revert(0, 0) }
|
// if slt(sub(_1, _2), 64) { revert(0, 0) }
|
||||||
// sstore(0, and(calldataload(_2), 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF))
|
// sstore(0, and(calldataload(_2), sub(shl(160, 1), 1)))
|
||||||
// let x0, x1, x2, x3, x4 := abi_decode_tuple_t_addresst_uint256t_bytes_calldata_ptrt_enum$_Operation_$1949(mload(7), mload(8))
|
// let x0, x1, x2, x3, x4 := abi_decode_tuple_t_addresst_uint256t_bytes_calldata_ptrt_enum$_Operation_$1949(mload(7), mload(8))
|
||||||
// sstore(x1, x0)
|
// sstore(x1, x0)
|
||||||
// sstore(x3, x2)
|
// sstore(x3, x2)
|
||||||
@ -1087,7 +1089,7 @@
|
|||||||
// function abi_decode_tuple_t_addresst_uint256t_bytes_calldata_ptrt_enum$_Operation_$1949(headStart, dataEnd) -> value0, value1, value2, value3, value4
|
// function abi_decode_tuple_t_addresst_uint256t_bytes_calldata_ptrt_enum$_Operation_$1949(headStart, dataEnd) -> value0, value1, value2, value3, value4
|
||||||
// {
|
// {
|
||||||
// if slt(sub(dataEnd, headStart), 128) { revert(value4, value4) }
|
// if slt(sub(dataEnd, headStart), 128) { revert(value4, value4) }
|
||||||
// value0 := and(calldataload(headStart), 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF)
|
// value0 := and(calldataload(headStart), sub(shl(160, 1), 1))
|
||||||
// value1 := calldataload(add(headStart, 32))
|
// value1 := calldataload(add(headStart, 32))
|
||||||
// let offset := calldataload(add(headStart, 64))
|
// let offset := calldataload(add(headStart, 64))
|
||||||
// let _1 := 0xffffffffffffffff
|
// let _1 := 0xffffffffffffffff
|
||||||
@ -1107,7 +1109,7 @@
|
|||||||
// {
|
// {
|
||||||
// tail := add(headStart, 352)
|
// tail := add(headStart, 352)
|
||||||
// mstore(headStart, value0)
|
// mstore(headStart, value0)
|
||||||
// let _1 := 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF
|
// let _1 := sub(shl(160, 1), 1)
|
||||||
// mstore(add(headStart, 32), and(value1, _1))
|
// mstore(add(headStart, 32), and(value1, _1))
|
||||||
// mstore(add(headStart, 64), value2)
|
// mstore(add(headStart, 64), value2)
|
||||||
// mstore(add(headStart, 96), value3)
|
// mstore(add(headStart, 96), value3)
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
// We require constantinople because numbers are represented using shifts.
|
||||||
{
|
{
|
||||||
let x := abi_encode_t_array$_t_array$_t_contract$_C_$55_$3_memory_$dyn_memory_ptr_to_t_array$_t_array$_t_address_$3_memory_$dyn_memory_ptr(mload(0), 0x20)
|
let x := abi_encode_t_array$_t_array$_t_contract$_C_$55_$3_memory_$dyn_memory_ptr_to_t_array$_t_array$_t_address_$3_memory_$dyn_memory_ptr(mload(0), 0x20)
|
||||||
let a, b, c, d := abi_decode_tuple_t_uint256t_uint256t_array$_t_uint256_$dyn_memory_ptrt_array$_t_array$_t_uint256_$2_memory_$dyn_memory_ptr(mload(0x20), mload(0x40))
|
let a, b, c, d := abi_decode_tuple_t_uint256t_uint256t_array$_t_uint256_$dyn_memory_ptrt_array$_t_array$_t_uint256_$2_memory_$dyn_memory_ptr(mload(0x20), mload(0x40))
|
||||||
@ -456,6 +457,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ====
|
// ====
|
||||||
|
// EVMVersion: >=constantinople
|
||||||
// step: fullSuite
|
// step: fullSuite
|
||||||
// ----
|
// ----
|
||||||
// {
|
// {
|
||||||
@ -546,7 +548,7 @@
|
|||||||
// let i := 0
|
// let i := 0
|
||||||
// for { } lt(i, 0x3) { i := add(i, 1) }
|
// for { } lt(i, 0x3) { i := add(i, 1) }
|
||||||
// {
|
// {
|
||||||
// mstore(pos, and(mload(srcPtr), 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF))
|
// mstore(pos, and(mload(srcPtr), sub(shl(160, 1), 1)))
|
||||||
// srcPtr := add(srcPtr, 0x20)
|
// srcPtr := add(srcPtr, 0x20)
|
||||||
// pos := add(pos, 0x20)
|
// pos := add(pos, 0x20)
|
||||||
// }
|
// }
|
||||||
|
@ -245,7 +245,7 @@
|
|||||||
// mstore(0x00, 404)
|
// mstore(0x00, 404)
|
||||||
// revert(0x00, 0x20)
|
// revert(0x00, 0x20)
|
||||||
// }
|
// }
|
||||||
// let kn := calldataload(add(calldatasize(), 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff40))
|
// let kn := calldataload(add(calldatasize(), not(191)))
|
||||||
// mstore(0x2a0, caller())
|
// mstore(0x2a0, caller())
|
||||||
// mstore(0x2c0, kn)
|
// mstore(0x2c0, kn)
|
||||||
// mstore(0x2e0, m)
|
// mstore(0x2e0, m)
|
||||||
@ -314,7 +314,7 @@
|
|||||||
// b := add(b, _3)
|
// b := add(b, _3)
|
||||||
// }
|
// }
|
||||||
// if lt(m, n) { validatePairing(0x64) }
|
// if lt(m, n) { validatePairing(0x64) }
|
||||||
// if iszero(eq(mod(keccak256(0x2a0, add(b, 0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffd60)), gen_order), challenge))
|
// if iszero(eq(mod(keccak256(0x2a0, add(b, not(671))), gen_order), challenge))
|
||||||
// {
|
// {
|
||||||
// mstore(i_1, 404)
|
// mstore(i_1, 404)
|
||||||
// revert(i_1, 0x20)
|
// revert(i_1, 0x20)
|
||||||
|
@ -37,20 +37,6 @@ using namespace yul::test;
|
|||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
|
|
||||||
u256 exp256(u256 _base, u256 _exponent)
|
|
||||||
{
|
|
||||||
using boost::multiprecision::limb_type;
|
|
||||||
u256 result = 1;
|
|
||||||
while (_exponent)
|
|
||||||
{
|
|
||||||
if (boost::multiprecision::bit_test(_exponent, 0))
|
|
||||||
result *= _base;
|
|
||||||
_base *= _base;
|
|
||||||
_exponent >>= 1;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Reads 32 bytes from @a _data at position @a _offset bytes while
|
/// Reads 32 bytes from @a _data at position @a _offset bytes while
|
||||||
/// interpreting @a _data to be padded with an infinite number of zero
|
/// interpreting @a _data to be padded with an infinite number of zero
|
||||||
/// bytes beyond its end.
|
/// bytes beyond its end.
|
||||||
|
Loading…
Reference in New Issue
Block a user