mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
493 lines
15 KiB
C++
493 lines
15 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/>.
|
|
*/
|
|
// SPDX-License-Identifier: GPL-3.0
|
|
/**
|
|
* Yul dialects for EVM.
|
|
*/
|
|
|
|
#include <libyul/backends/evm/EVMDialect.h>
|
|
|
|
#include <libevmasm/Instruction.h>
|
|
#include <libevmasm/SemanticInformation.h>
|
|
#include <liblangutil/Exceptions.h>
|
|
#include <libsolutil/StringUtils.h>
|
|
#include <libyul/AST.h>
|
|
#include <libyul/AsmAnalysisInfo.h>
|
|
#include <libyul/AsmParser.h>
|
|
#include <libyul/Exceptions.h>
|
|
#include <libyul/Object.h>
|
|
#include <libyul/Utilities.h>
|
|
#include <libyul/backends/evm/AbstractAssembly.h>
|
|
|
|
#include <range/v3/view/reverse.hpp>
|
|
#include <range/v3/view/tail.hpp>
|
|
|
|
#include <regex>
|
|
|
|
using namespace std;
|
|
using namespace solidity;
|
|
using namespace solidity::yul;
|
|
using namespace solidity::util;
|
|
|
|
namespace
|
|
{
|
|
|
|
pair<YulString, BuiltinFunctionForEVM> createEVMFunction(
|
|
string const& _name,
|
|
evmasm::Instruction _instruction
|
|
)
|
|
{
|
|
evmasm::InstructionInfo info = evmasm::instructionInfo(_instruction);
|
|
BuiltinFunctionForEVM f;
|
|
f.name = YulString{_name};
|
|
f.parameters.resize(static_cast<size_t>(info.args));
|
|
f.returns.resize(static_cast<size_t>(info.ret));
|
|
f.sideEffects = EVMDialect::sideEffectsOfInstruction(_instruction);
|
|
if (evmasm::SemanticInformation::terminatesControlFlow(_instruction))
|
|
{
|
|
f.controlFlowSideEffects.canContinue = false;
|
|
if (evmasm::SemanticInformation::reverts(_instruction))
|
|
{
|
|
f.controlFlowSideEffects.canTerminate = false;
|
|
f.controlFlowSideEffects.canRevert = true;
|
|
}
|
|
else
|
|
{
|
|
f.controlFlowSideEffects.canTerminate = true;
|
|
f.controlFlowSideEffects.canRevert = false;
|
|
}
|
|
}
|
|
f.isMSize = _instruction == evmasm::Instruction::MSIZE;
|
|
f.literalArguments.clear();
|
|
f.instruction = _instruction;
|
|
f.generateCode = [_instruction](
|
|
FunctionCall const&,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
_assembly.appendInstruction(_instruction);
|
|
};
|
|
|
|
YulString name = f.name;
|
|
return {name, move(f)};
|
|
}
|
|
|
|
pair<YulString, BuiltinFunctionForEVM> createFunction(
|
|
string _name,
|
|
size_t _params,
|
|
size_t _returns,
|
|
SideEffects _sideEffects,
|
|
vector<optional<LiteralKind>> _literalArguments,
|
|
std::function<void(FunctionCall const&, AbstractAssembly&, BuiltinContext&)> _generateCode
|
|
)
|
|
{
|
|
yulAssert(_literalArguments.size() == _params || _literalArguments.empty(), "");
|
|
|
|
YulString name{std::move(_name)};
|
|
BuiltinFunctionForEVM f;
|
|
f.name = name;
|
|
f.parameters.resize(_params);
|
|
f.returns.resize(_returns);
|
|
f.sideEffects = std::move(_sideEffects);
|
|
f.literalArguments = std::move(_literalArguments);
|
|
f.isMSize = false;
|
|
f.instruction = {};
|
|
f.generateCode = std::move(_generateCode);
|
|
return {name, f};
|
|
}
|
|
|
|
set<YulString> createReservedIdentifiers(langutil::EVMVersion _evmVersion)
|
|
{
|
|
// TODO remove this in 0.9.0. We allow creating functions or identifiers in Yul with the name
|
|
// basefee for VMs before london.
|
|
auto baseFeeException = [&](evmasm::Instruction _instr) -> bool
|
|
{
|
|
return _instr == evmasm::Instruction::BASEFEE && _evmVersion < langutil::EVMVersion::london();
|
|
};
|
|
|
|
set<YulString> reserved;
|
|
for (auto const& instr: evmasm::c_instructions)
|
|
{
|
|
string name = toLower(instr.first);
|
|
if (!baseFeeException(instr.second))
|
|
reserved.emplace(name);
|
|
}
|
|
reserved += vector<YulString>{
|
|
"linkersymbol"_yulstring,
|
|
"datasize"_yulstring,
|
|
"dataoffset"_yulstring,
|
|
"datacopy"_yulstring,
|
|
"setimmutable"_yulstring,
|
|
"loadimmutable"_yulstring,
|
|
};
|
|
return reserved;
|
|
}
|
|
|
|
map<YulString, BuiltinFunctionForEVM> createBuiltins(langutil::EVMVersion _evmVersion, bool _objectAccess)
|
|
{
|
|
map<YulString, BuiltinFunctionForEVM> builtins;
|
|
for (auto const& instr: evmasm::c_instructions)
|
|
{
|
|
string name = toLower(instr.first);
|
|
auto const opcode = instr.second;
|
|
|
|
if (
|
|
!evmasm::isDupInstruction(opcode) &&
|
|
!evmasm::isSwapInstruction(opcode) &&
|
|
!evmasm::isPushInstruction(opcode) &&
|
|
opcode != evmasm::Instruction::JUMP &&
|
|
opcode != evmasm::Instruction::JUMPI &&
|
|
opcode != evmasm::Instruction::JUMPDEST &&
|
|
_evmVersion.hasOpcode(opcode)
|
|
)
|
|
builtins.emplace(createEVMFunction(name, opcode));
|
|
}
|
|
|
|
if (_objectAccess)
|
|
{
|
|
builtins.emplace(createFunction("linkersymbol", 1, 1, SideEffects{}, {LiteralKind::String}, [](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
Expression const& arg = _call.arguments.front();
|
|
_assembly.appendLinkerSymbol(std::get<Literal>(arg).value.str());
|
|
}));
|
|
|
|
builtins.emplace(createFunction(
|
|
"memoryguard",
|
|
1,
|
|
1,
|
|
SideEffects{},
|
|
{LiteralKind::Number},
|
|
[](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
Literal const* literal = get_if<Literal>(&_call.arguments.front());
|
|
yulAssert(literal, "");
|
|
_assembly.appendConstant(valueOfLiteral(*literal));
|
|
})
|
|
);
|
|
|
|
builtins.emplace(createFunction("datasize", 1, 1, SideEffects{}, {LiteralKind::String}, [](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext& _context
|
|
) {
|
|
yulAssert(_context.currentObject, "No object available.");
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
Expression const& arg = _call.arguments.front();
|
|
YulString dataName = std::get<Literal>(arg).value;
|
|
if (_context.currentObject->name == dataName)
|
|
_assembly.appendAssemblySize();
|
|
else
|
|
{
|
|
vector<size_t> subIdPath =
|
|
_context.subIDs.count(dataName) == 0 ?
|
|
_context.currentObject->pathToSubObject(dataName) :
|
|
vector<size_t>{_context.subIDs.at(dataName)};
|
|
yulAssert(!subIdPath.empty(), "Could not find assembly object <" + dataName.str() + ">.");
|
|
_assembly.appendDataSize(subIdPath);
|
|
}
|
|
}));
|
|
builtins.emplace(createFunction("dataoffset", 1, 1, SideEffects{}, {LiteralKind::String}, [](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext& _context
|
|
) {
|
|
yulAssert(_context.currentObject, "No object available.");
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
Expression const& arg = _call.arguments.front();
|
|
YulString dataName = std::get<Literal>(arg).value;
|
|
if (_context.currentObject->name == dataName)
|
|
_assembly.appendConstant(0);
|
|
else
|
|
{
|
|
vector<size_t> subIdPath =
|
|
_context.subIDs.count(dataName) == 0 ?
|
|
_context.currentObject->pathToSubObject(dataName) :
|
|
vector<size_t>{_context.subIDs.at(dataName)};
|
|
yulAssert(!subIdPath.empty(), "Could not find assembly object <" + dataName.str() + ">.");
|
|
_assembly.appendDataOffset(subIdPath);
|
|
}
|
|
}));
|
|
builtins.emplace(createFunction(
|
|
"datacopy",
|
|
3,
|
|
0,
|
|
SideEffects{false, true, false, false, true, SideEffects::None, SideEffects::None, SideEffects::Write},
|
|
{},
|
|
[](
|
|
FunctionCall const&,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
_assembly.appendInstruction(evmasm::Instruction::CODECOPY);
|
|
}
|
|
));
|
|
builtins.emplace(createFunction(
|
|
"setimmutable",
|
|
3,
|
|
0,
|
|
SideEffects{false, false, false, false, true, SideEffects::None, SideEffects::None, SideEffects::Write},
|
|
{std::nullopt, LiteralKind::String, std::nullopt},
|
|
[](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
yulAssert(_call.arguments.size() == 3, "");
|
|
YulString identifier = std::get<Literal>(_call.arguments[1]).value;
|
|
_assembly.appendImmutableAssignment(identifier.str());
|
|
}
|
|
));
|
|
builtins.emplace(createFunction(
|
|
"loadimmutable",
|
|
1,
|
|
1,
|
|
SideEffects{},
|
|
{LiteralKind::String},
|
|
[](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
_assembly.appendImmutable(std::get<Literal>(_call.arguments.front()).value.str());
|
|
}
|
|
));
|
|
}
|
|
return builtins;
|
|
}
|
|
|
|
regex const& verbatimPattern()
|
|
{
|
|
regex static const pattern{"verbatim_([1-9]?[0-9])i_([1-9]?[0-9])o"};
|
|
return pattern;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
EVMDialect::EVMDialect(langutil::EVMVersion _evmVersion, bool _objectAccess):
|
|
m_objectAccess(_objectAccess),
|
|
m_evmVersion(_evmVersion),
|
|
m_functions(createBuiltins(_evmVersion, _objectAccess)),
|
|
m_reserved(createReservedIdentifiers(_evmVersion))
|
|
{
|
|
}
|
|
|
|
BuiltinFunctionForEVM const* EVMDialect::builtin(YulString _name) const
|
|
{
|
|
if (m_objectAccess)
|
|
{
|
|
smatch match;
|
|
if (regex_match(_name.str(), match, verbatimPattern()))
|
|
return verbatimFunction(stoul(match[1]), stoul(match[2]));
|
|
}
|
|
auto it = m_functions.find(_name);
|
|
if (it != m_functions.end())
|
|
return &it->second;
|
|
else
|
|
return nullptr;
|
|
}
|
|
|
|
bool EVMDialect::reservedIdentifier(YulString _name) const
|
|
{
|
|
if (m_objectAccess)
|
|
if (_name.str().substr(0, "verbatim"s.size()) == "verbatim")
|
|
return true;
|
|
return m_reserved.count(_name) != 0;
|
|
}
|
|
|
|
EVMDialect const& EVMDialect::strictAssemblyForEVM(langutil::EVMVersion _version)
|
|
{
|
|
static map<langutil::EVMVersion, unique_ptr<EVMDialect const>> dialects;
|
|
static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }};
|
|
if (!dialects[_version])
|
|
dialects[_version] = make_unique<EVMDialect>(_version, false);
|
|
return *dialects[_version];
|
|
}
|
|
|
|
EVMDialect const& EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion _version)
|
|
{
|
|
static map<langutil::EVMVersion, unique_ptr<EVMDialect const>> dialects;
|
|
static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }};
|
|
if (!dialects[_version])
|
|
dialects[_version] = make_unique<EVMDialect>(_version, true);
|
|
return *dialects[_version];
|
|
}
|
|
|
|
SideEffects EVMDialect::sideEffectsOfInstruction(evmasm::Instruction _instruction)
|
|
{
|
|
auto translate = [](evmasm::SemanticInformation::Effect _e) -> SideEffects::Effect
|
|
{
|
|
return static_cast<SideEffects::Effect>(_e);
|
|
};
|
|
|
|
return SideEffects{
|
|
evmasm::SemanticInformation::movable(_instruction),
|
|
evmasm::SemanticInformation::movableApartFromEffects(_instruction),
|
|
evmasm::SemanticInformation::canBeRemoved(_instruction),
|
|
evmasm::SemanticInformation::canBeRemovedIfNoMSize(_instruction),
|
|
true, // cannotLoop
|
|
translate(evmasm::SemanticInformation::otherState(_instruction)),
|
|
translate(evmasm::SemanticInformation::storage(_instruction)),
|
|
translate(evmasm::SemanticInformation::memory(_instruction)),
|
|
};
|
|
}
|
|
|
|
BuiltinFunctionForEVM const* EVMDialect::verbatimFunction(size_t _arguments, size_t _returnVariables) const
|
|
{
|
|
pair<size_t, size_t> key{_arguments, _returnVariables};
|
|
shared_ptr<BuiltinFunctionForEVM const>& function = m_verbatimFunctions[key];
|
|
if (!function)
|
|
{
|
|
BuiltinFunctionForEVM builtinFunction = createFunction(
|
|
"verbatim_" + to_string(_arguments) + "i_" + to_string(_returnVariables) + "o",
|
|
1 + _arguments,
|
|
_returnVariables,
|
|
SideEffects::worst(),
|
|
vector<optional<LiteralKind>>{LiteralKind::String} + vector<optional<LiteralKind>>(_arguments),
|
|
[=](
|
|
FunctionCall const& _call,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
yulAssert(_call.arguments.size() == (1 + _arguments), "");
|
|
Expression const& bytecode = _call.arguments.front();
|
|
|
|
_assembly.appendVerbatim(
|
|
asBytes(std::get<Literal>(bytecode).value.str()),
|
|
_arguments,
|
|
_returnVariables
|
|
);
|
|
}
|
|
).second;
|
|
builtinFunction.isMSize = true;
|
|
function = make_shared<BuiltinFunctionForEVM const>(std::move(builtinFunction));
|
|
}
|
|
return function.get();
|
|
}
|
|
|
|
EVMDialectTyped::EVMDialectTyped(langutil::EVMVersion _evmVersion, bool _objectAccess):
|
|
EVMDialect(_evmVersion, _objectAccess)
|
|
{
|
|
defaultType = "u256"_yulstring;
|
|
boolType = "bool"_yulstring;
|
|
types = {defaultType, boolType};
|
|
|
|
// Set all types to ``defaultType``
|
|
for (auto& fun: m_functions)
|
|
{
|
|
for (auto& p: fun.second.parameters)
|
|
p = defaultType;
|
|
for (auto& r: fun.second.returns)
|
|
r = defaultType;
|
|
}
|
|
|
|
m_functions["lt"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["gt"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["slt"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["sgt"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["eq"_yulstring].returns = {"bool"_yulstring};
|
|
|
|
// "not" and "bitnot" replace "iszero" and "not"
|
|
m_functions["bitnot"_yulstring] = m_functions["not"_yulstring];
|
|
m_functions["bitnot"_yulstring].name = "bitnot"_yulstring;
|
|
m_functions["not"_yulstring] = m_functions["iszero"_yulstring];
|
|
m_functions["not"_yulstring].name = "not"_yulstring;
|
|
m_functions["not"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["not"_yulstring].parameters = {"bool"_yulstring};
|
|
m_functions.erase("iszero"_yulstring);
|
|
|
|
m_functions["bitand"_yulstring] = m_functions["and"_yulstring];
|
|
m_functions["bitand"_yulstring].name = "bitand"_yulstring;
|
|
m_functions["bitor"_yulstring] = m_functions["or"_yulstring];
|
|
m_functions["bitor"_yulstring].name = "bitor"_yulstring;
|
|
m_functions["bitxor"_yulstring] = m_functions["xor"_yulstring];
|
|
m_functions["bitxor"_yulstring].name = "bitxor"_yulstring;
|
|
m_functions["and"_yulstring].parameters = {"bool"_yulstring, "bool"_yulstring};
|
|
m_functions["and"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["or"_yulstring].parameters = {"bool"_yulstring, "bool"_yulstring};
|
|
m_functions["or"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["xor"_yulstring].parameters = {"bool"_yulstring, "bool"_yulstring};
|
|
m_functions["xor"_yulstring].returns = {"bool"_yulstring};
|
|
m_functions["popbool"_yulstring] = m_functions["pop"_yulstring];
|
|
m_functions["popbool"_yulstring].name = "popbool"_yulstring;
|
|
m_functions["popbool"_yulstring].parameters = {"bool"_yulstring};
|
|
m_functions.insert(createFunction("bool_to_u256", 1, 1, {}, {}, [](
|
|
FunctionCall const&,
|
|
AbstractAssembly&,
|
|
BuiltinContext&
|
|
) {}));
|
|
m_functions["bool_to_u256"_yulstring].parameters = {"bool"_yulstring};
|
|
m_functions["bool_to_u256"_yulstring].returns = {"u256"_yulstring};
|
|
m_functions.insert(createFunction("u256_to_bool", 1, 1, {}, {}, [](
|
|
FunctionCall const&,
|
|
AbstractAssembly& _assembly,
|
|
BuiltinContext&
|
|
) {
|
|
// TODO this should use a Panic.
|
|
// A value larger than 1 causes an invalid instruction.
|
|
_assembly.appendConstant(2);
|
|
_assembly.appendInstruction(evmasm::Instruction::DUP2);
|
|
_assembly.appendInstruction(evmasm::Instruction::LT);
|
|
AbstractAssembly::LabelID inRange = _assembly.newLabelId();
|
|
_assembly.appendJumpToIf(inRange);
|
|
_assembly.appendInstruction(evmasm::Instruction::INVALID);
|
|
_assembly.appendLabel(inRange);
|
|
}));
|
|
m_functions["u256_to_bool"_yulstring].parameters = {"u256"_yulstring};
|
|
m_functions["u256_to_bool"_yulstring].returns = {"bool"_yulstring};
|
|
}
|
|
|
|
BuiltinFunctionForEVM const* EVMDialectTyped::discardFunction(YulString _type) const
|
|
{
|
|
if (_type == "bool"_yulstring)
|
|
return builtin("popbool"_yulstring);
|
|
else
|
|
{
|
|
yulAssert(_type == defaultType, "");
|
|
return builtin("pop"_yulstring);
|
|
}
|
|
}
|
|
|
|
BuiltinFunctionForEVM const* EVMDialectTyped::equalityFunction(YulString _type) const
|
|
{
|
|
if (_type == "bool"_yulstring)
|
|
return nullptr;
|
|
else
|
|
{
|
|
yulAssert(_type == defaultType, "");
|
|
return builtin("eq"_yulstring);
|
|
}
|
|
}
|
|
|
|
EVMDialectTyped const& EVMDialectTyped::instance(langutil::EVMVersion _version)
|
|
{
|
|
static map<langutil::EVMVersion, unique_ptr<EVMDialectTyped const>> dialects;
|
|
static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }};
|
|
if (!dialects[_version])
|
|
dialects[_version] = make_unique<EVMDialectTyped>(_version, true);
|
|
return *dialects[_version];
|
|
}
|