solidity/libyul/backends/evm/EVMObjectCompiler.cpp

98 lines
3.0 KiB
C++
Raw Normal View History

2018-12-04 17:57:32 +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/>.
*/
// SPDX-License-Identifier: GPL-3.0
2018-12-04 17:57:32 +00:00
/**
* Compiler that transforms Yul Objects to EVM bytecode objects.
*/
#include <libyul/backends/evm/EVMObjectCompiler.h>
#include <libyul/backends/evm/EVMCodeTransform.h>
2018-12-06 23:56:16 +00:00
#include <libyul/backends/evm/EVMDialect.h>
#include <libyul/backends/evm/OptimizedEVMCodeTransform.h>
2018-12-06 23:56:16 +00:00
2018-12-04 17:57:32 +00:00
#include <libyul/Object.h>
#include <libyul/Exceptions.h>
2019-12-11 16:31:36 +00:00
using namespace solidity::yul;
2018-12-04 17:57:32 +00:00
using namespace std;
2021-03-09 11:46:33 +00:00
void EVMObjectCompiler::compile(Object& _object, AbstractAssembly& _assembly, EVMDialect const& _dialect, bool _optimize)
2018-12-04 17:57:32 +00:00
{
2021-03-09 11:46:33 +00:00
EVMObjectCompiler compiler(_assembly, _dialect);
compiler.run(_object, _optimize);
2018-12-04 17:57:32 +00:00
}
void EVMObjectCompiler::run(Object& _object, bool _optimize)
2018-12-04 17:57:32 +00:00
{
BuiltinContext context;
context.currentObject = &_object;
2018-12-04 17:57:32 +00:00
for (auto const& subNode: _object.subObjects)
if (auto* subObject = dynamic_cast<Object*>(subNode.get()))
2018-12-04 17:57:32 +00:00
{
auto subAssemblyAndID = m_assembly.createSubAssembly(subObject->name.str());
context.subIDs[subObject->name] = subAssemblyAndID.second;
subObject->subId = subAssemblyAndID.second;
2021-03-09 11:46:33 +00:00
compile(*subObject, *subAssemblyAndID.first, m_dialect, _optimize);
2018-12-04 17:57:32 +00:00
}
else
{
Data const& data = dynamic_cast<Data const&>(*subNode);
2021-06-08 14:35:37 +00:00
// Special handling of metadata.
if (data.name.str() == Object::metadataName())
m_assembly.appendToAuxiliaryData(data.data);
else
context.subIDs[data.name] = m_assembly.appendData(data.data);
2018-12-04 17:57:32 +00:00
}
yulAssert(_object.analysisInfo, "No analysis info.");
yulAssert(_object.code, "No code.");
if (_optimize && m_dialect.evmVersion().canOverchargeGasForCall())
{
2021-11-01 13:19:33 +00:00
auto stackErrors = OptimizedEVMCodeTransform::run(
m_assembly,
*_object.analysisInfo,
*_object.code,
m_dialect,
context,
OptimizedEVMCodeTransform::UseNamedLabels::ForFirstFunctionOfEachName
);
if (!stackErrors.empty())
BOOST_THROW_EXCEPTION(stackErrors.front());
}
else
{
// We do not catch and re-throw the stack too deep exception here because it is a YulException,
// which should be native to this part of the code.
CodeTransform transform{
m_assembly,
*_object.analysisInfo,
*_object.code,
m_dialect,
context,
_optimize,
{},
CodeTransform::UseNamedLabels::ForFirstFunctionOfEachName
};
transform(*_object.code);
if (!transform.stackErrors().empty())
BOOST_THROW_EXCEPTION(transform.stackErrors().front());
}
2018-12-04 17:57:32 +00:00
}