2018-12-06 23:56:16 +00:00
|
|
|
/*
|
|
|
|
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/>.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Yul dialects for EVM.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <libyul/backends/evm/EVMDialect.h>
|
|
|
|
|
|
|
|
#include <libyul/AsmAnalysisInfo.h>
|
|
|
|
#include <libyul/AsmData.h>
|
|
|
|
#include <libyul/Object.h>
|
|
|
|
#include <libyul/backends/evm/AbstractAssembly.h>
|
|
|
|
|
|
|
|
#include <liblangutil/Exceptions.h>
|
|
|
|
|
|
|
|
#include <libyul/Exceptions.h>
|
|
|
|
|
|
|
|
#include <boost/range/adaptor/reversed.hpp>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace dev;
|
|
|
|
using namespace yul;
|
|
|
|
using namespace dev::solidity;
|
|
|
|
|
|
|
|
|
|
|
|
EVMDialect::EVMDialect(AsmFlavour _flavour, bool _objectAccess):
|
2018-12-12 13:51:22 +00:00
|
|
|
Dialect{_flavour}, m_objectAccess(_objectAccess)
|
2018-12-06 23:56:16 +00:00
|
|
|
{
|
|
|
|
// The EVM instructions will be moved to builtins at some point.
|
|
|
|
if (!m_objectAccess)
|
|
|
|
return;
|
|
|
|
|
2018-12-20 16:22:17 +00:00
|
|
|
addFunction("datasize", 1, 1, true, true, [this](
|
2018-12-06 23:56:16 +00:00
|
|
|
FunctionCall const& _call,
|
|
|
|
AbstractAssembly& _assembly,
|
|
|
|
std::function<void()>
|
|
|
|
) {
|
|
|
|
yulAssert(m_currentObject, "No object available.");
|
|
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
|
|
Expression const& arg = _call.arguments.front();
|
|
|
|
YulString dataName = boost::get<Literal>(arg).value;
|
|
|
|
if (m_currentObject->name == dataName)
|
|
|
|
_assembly.appendAssemblySize();
|
|
|
|
else
|
|
|
|
_assembly.appendDataSize(m_subIDs.at(dataName));
|
|
|
|
});
|
2018-12-20 16:22:17 +00:00
|
|
|
addFunction("dataoffset", 1, 1, true, true, [this](
|
2018-12-06 23:56:16 +00:00
|
|
|
FunctionCall const& _call,
|
|
|
|
AbstractAssembly& _assembly,
|
|
|
|
std::function<void()>
|
|
|
|
) {
|
|
|
|
yulAssert(m_currentObject, "No object available.");
|
|
|
|
yulAssert(_call.arguments.size() == 1, "");
|
|
|
|
Expression const& arg = _call.arguments.front();
|
|
|
|
YulString dataName = boost::get<Literal>(arg).value;
|
|
|
|
if (m_currentObject->name == dataName)
|
|
|
|
_assembly.appendConstant(0);
|
|
|
|
else
|
|
|
|
_assembly.appendDataOffset(m_subIDs.at(dataName));
|
|
|
|
});
|
2018-12-20 16:22:17 +00:00
|
|
|
addFunction("datacopy", 3, 0, false, false, [](
|
2018-12-06 23:56:16 +00:00
|
|
|
FunctionCall const&,
|
|
|
|
AbstractAssembly& _assembly,
|
|
|
|
std::function<void()> _visitArguments
|
|
|
|
) {
|
|
|
|
_visitArguments();
|
|
|
|
_assembly.appendInstruction(solidity::Instruction::CODECOPY);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
BuiltinFunctionForEVM const* EVMDialect::builtin(YulString _name) const
|
|
|
|
{
|
|
|
|
auto it = m_functions.find(_name);
|
|
|
|
if (it != m_functions.end())
|
|
|
|
return &it->second;
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
shared_ptr<EVMDialect> EVMDialect::looseAssemblyForEVM()
|
|
|
|
{
|
|
|
|
return make_shared<EVMDialect>(AsmFlavour::Loose, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
shared_ptr<EVMDialect> EVMDialect::strictAssemblyForEVM()
|
|
|
|
{
|
|
|
|
return make_shared<EVMDialect>(AsmFlavour::Strict, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
shared_ptr<EVMDialect> EVMDialect::strictAssemblyForEVMObjects()
|
|
|
|
{
|
|
|
|
return make_shared<EVMDialect>(AsmFlavour::Strict, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
shared_ptr<yul::EVMDialect> EVMDialect::yulForEVM()
|
|
|
|
{
|
|
|
|
return make_shared<EVMDialect>(AsmFlavour::Yul, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EVMDialect::setSubIDs(map<YulString, AbstractAssembly::SubID> _subIDs)
|
|
|
|
{
|
|
|
|
yulAssert(m_objectAccess, "Sub IDs set with dialect that does not support object access.");
|
|
|
|
m_subIDs = std::move(_subIDs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EVMDialect::setCurrentObject(Object const* _object)
|
|
|
|
{
|
|
|
|
yulAssert(m_objectAccess, "Current object set with dialect that does not support object access.");
|
|
|
|
m_currentObject = _object;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EVMDialect::addFunction(
|
|
|
|
string _name,
|
|
|
|
size_t _params,
|
|
|
|
size_t _returns,
|
|
|
|
bool _movable,
|
2018-12-20 16:22:17 +00:00
|
|
|
bool _literalArguments,
|
2018-12-06 23:56:16 +00:00
|
|
|
std::function<void(FunctionCall const&, AbstractAssembly&, std::function<void()>)> _generateCode
|
|
|
|
)
|
|
|
|
{
|
|
|
|
YulString name{std::move(_name)};
|
|
|
|
BuiltinFunctionForEVM& f = m_functions[name];
|
|
|
|
f.name = name;
|
|
|
|
f.parameters.resize(_params);
|
|
|
|
f.returns.resize(_returns);
|
|
|
|
f.movable = _movable;
|
2018-12-20 16:22:17 +00:00
|
|
|
f.literalArguments = _literalArguments;
|
2018-12-06 23:56:16 +00:00
|
|
|
f.generateCode = std::move(_generateCode);
|
|
|
|
}
|