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/>.
|
|
|
|
*/
|
2020-07-17 14:54:12 +00:00
|
|
|
// 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>
|
2021-08-12 22:37:23 +00:00
|
|
|
#include <libyul/backends/evm/OptimizedEVMCodeTransform.h>
|
2018-12-06 23:56:16 +00:00
|
|
|
|
2022-02-03 11:37:40 +00:00
|
|
|
#include <libyul/optimiser/FunctionCallFinder.h>
|
|
|
|
|
2018-12-04 17:57:32 +00:00
|
|
|
#include <libyul/Object.h>
|
|
|
|
#include <libyul/Exceptions.h>
|
|
|
|
|
2022-03-07 11:49:23 +00:00
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
|
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);
|
2018-09-25 02:47:25 +00:00
|
|
|
compiler.run(_object, _optimize);
|
2018-12-04 17:57:32 +00:00
|
|
|
}
|
|
|
|
|
2018-09-25 02:47:25 +00:00
|
|
|
void EVMObjectCompiler::run(Object& _object, bool _optimize)
|
2018-12-04 17:57:32 +00:00
|
|
|
{
|
2019-05-15 21:40:30 +00:00
|
|
|
BuiltinContext context;
|
|
|
|
context.currentObject = &_object;
|
2018-12-04 17:57:32 +00:00
|
|
|
|
2021-02-03 10:47:16 +00:00
|
|
|
|
2020-06-17 09:17:35 +00:00
|
|
|
for (auto const& subNode: _object.subObjects)
|
|
|
|
if (auto* subObject = dynamic_cast<Object*>(subNode.get()))
|
2018-12-04 17:57:32 +00:00
|
|
|
{
|
2022-03-07 11:49:23 +00:00
|
|
|
bool isCreation = !boost::ends_with(subObject->name.str(), "_deployed");
|
|
|
|
auto subAssemblyAndID = m_assembly.createSubAssembly(isCreation, subObject->name.str());
|
2019-05-15 21:40:30 +00:00
|
|
|
context.subIDs[subObject->name] = subAssemblyAndID.second;
|
2020-06-17 09:17:35 +00:00
|
|
|
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.");
|
2018-09-25 02:47:25 +00:00
|
|
|
yulAssert(_object.code, "No code.");
|
2021-08-12 22:37:23 +00:00
|
|
|
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
|
|
|
|
);
|
2021-08-12 22:37:23 +00:00
|
|
|
if (!stackErrors.empty())
|
2022-02-03 11:37:40 +00:00
|
|
|
{
|
|
|
|
vector<FunctionCall*> memoryGuardCalls = FunctionCallFinder::run(
|
|
|
|
*_object.code,
|
|
|
|
"memoryguard"_yulstring
|
|
|
|
);
|
|
|
|
auto stackError = stackErrors.front();
|
|
|
|
string msg = stackError.comment() ? *stackError.comment() : "";
|
|
|
|
if (memoryGuardCalls.empty())
|
|
|
|
msg += "\nNo memoryguard was present. "
|
|
|
|
"Consider using memory-safe assembly only and annotating it via "
|
2022-03-14 10:54:09 +00:00
|
|
|
"'assembly (\"memory-safe\") { ... }'.";
|
2022-02-03 11:37:40 +00:00
|
|
|
else
|
|
|
|
msg += "\nmemoryguard was present.";
|
|
|
|
stackError << util::errinfo_comment(msg);
|
|
|
|
BOOST_THROW_EXCEPTION(stackError);
|
|
|
|
}
|
2021-08-12 22:37:23 +00:00
|
|
|
}
|
|
|
|
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
|
|
|
}
|