2019-03-04 22:26:46 +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
|
2019-03-04 22:26:46 +00:00
|
|
|
/**
|
|
|
|
* Class that contains contextual information during IR generation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <libsolidity/codegen/ir/IRGenerationContext.h>
|
|
|
|
|
2019-04-02 10:37:48 +00:00
|
|
|
#include <libsolidity/codegen/YulUtilFunctions.h>
|
2020-04-09 19:59:17 +00:00
|
|
|
#include <libsolidity/codegen/ABIFunctions.h>
|
2020-04-02 18:06:52 +00:00
|
|
|
#include <libsolidity/codegen/CompilerUtils.h>
|
2019-03-04 22:26:46 +00:00
|
|
|
#include <libsolidity/ast/AST.h>
|
2020-03-02 20:42:46 +00:00
|
|
|
#include <libsolidity/ast/TypeProvider.h>
|
2019-03-04 22:26:46 +00:00
|
|
|
|
2020-01-06 10:52:23 +00:00
|
|
|
#include <libsolutil/Whiskers.h>
|
|
|
|
#include <libsolutil/StringUtils.h>
|
2019-04-02 10:37:48 +00:00
|
|
|
|
2021-03-31 18:03:04 +00:00
|
|
|
#include <range/v3/view/map.hpp>
|
2020-05-19 19:50:22 +00:00
|
|
|
|
2019-03-04 22:26:46 +00:00
|
|
|
using namespace std;
|
2019-12-11 16:31:36 +00:00
|
|
|
using namespace solidity;
|
|
|
|
using namespace solidity::util;
|
|
|
|
using namespace solidity::frontend;
|
2019-03-04 22:26:46 +00:00
|
|
|
|
2020-04-10 17:16:44 +00:00
|
|
|
string IRGenerationContext::enqueueFunctionForCodeGeneration(FunctionDefinition const& _function)
|
|
|
|
{
|
2020-05-13 17:48:31 +00:00
|
|
|
string name = IRNames::function(_function);
|
2020-04-10 17:16:44 +00:00
|
|
|
|
|
|
|
if (!m_functions.contains(name))
|
|
|
|
m_functionGenerationQueue.insert(&_function);
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionDefinition const* IRGenerationContext::dequeueFunctionForCodeGeneration()
|
|
|
|
{
|
|
|
|
solAssert(!m_functionGenerationQueue.empty(), "");
|
|
|
|
|
|
|
|
FunctionDefinition const* result = *m_functionGenerationQueue.begin();
|
|
|
|
m_functionGenerationQueue.erase(m_functionGenerationQueue.begin());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-03-24 17:25:59 +00:00
|
|
|
ContractDefinition const& IRGenerationContext::mostDerivedContract() const
|
|
|
|
{
|
|
|
|
solAssert(m_mostDerivedContract, "Most derived contract requested but not set.");
|
|
|
|
return *m_mostDerivedContract;
|
|
|
|
}
|
|
|
|
|
2020-02-06 13:05:25 +00:00
|
|
|
IRVariable const& IRGenerationContext::addLocalVariable(VariableDeclaration const& _varDecl)
|
2019-03-04 22:26:46 +00:00
|
|
|
{
|
2020-02-06 13:05:25 +00:00
|
|
|
auto const& [it, didInsert] = m_localVariables.emplace(
|
|
|
|
std::make_pair(&_varDecl, IRVariable{_varDecl})
|
2019-03-04 22:26:46 +00:00
|
|
|
);
|
2020-02-06 13:05:25 +00:00
|
|
|
solAssert(didInsert, "Local variable added multiple times.");
|
|
|
|
return it->second;
|
2019-03-04 22:26:46 +00:00
|
|
|
}
|
2019-03-18 10:21:41 +00:00
|
|
|
|
2020-02-06 13:05:25 +00:00
|
|
|
IRVariable const& IRGenerationContext::localVariable(VariableDeclaration const& _varDecl)
|
2019-03-18 10:21:41 +00:00
|
|
|
{
|
|
|
|
solAssert(
|
|
|
|
m_localVariables.count(&_varDecl),
|
|
|
|
"Unknown variable: " + _varDecl.name()
|
|
|
|
);
|
2020-02-06 13:05:25 +00:00
|
|
|
return m_localVariables.at(&_varDecl);
|
2019-03-18 10:21:41 +00:00
|
|
|
}
|
|
|
|
|
2020-11-30 17:59:49 +00:00
|
|
|
void IRGenerationContext::resetLocalVariables()
|
|
|
|
{
|
|
|
|
m_localVariables.clear();
|
|
|
|
}
|
|
|
|
|
2020-04-02 18:06:52 +00:00
|
|
|
void IRGenerationContext::registerImmutableVariable(VariableDeclaration const& _variable)
|
|
|
|
{
|
|
|
|
solAssert(_variable.immutable(), "Attempted to register a non-immutable variable as immutable.");
|
|
|
|
solUnimplementedAssert(
|
|
|
|
_variable.annotation().type->isValueType(),
|
|
|
|
"Only immutable variables of value type are supported."
|
|
|
|
);
|
|
|
|
solAssert(m_reservedMemory.has_value(), "Reserved memory has already been reset.");
|
|
|
|
m_immutableVariables[&_variable] = CompilerUtils::generalPurposeMemoryStart + *m_reservedMemory;
|
|
|
|
solAssert(_variable.annotation().type->memoryHeadSize() == 32, "Memory writes might overlap.");
|
|
|
|
*m_reservedMemory += _variable.annotation().type->memoryHeadSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t IRGenerationContext::immutableMemoryOffset(VariableDeclaration const& _variable) const
|
|
|
|
{
|
|
|
|
solAssert(
|
|
|
|
m_immutableVariables.count(&_variable),
|
|
|
|
"Unknown immutable variable: " + _variable.name()
|
|
|
|
);
|
|
|
|
return m_immutableVariables.at(&_variable);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t IRGenerationContext::reservedMemory()
|
|
|
|
{
|
|
|
|
solAssert(m_reservedMemory.has_value(), "Reserved memory was used before.");
|
|
|
|
size_t reservedMemory = *m_reservedMemory;
|
|
|
|
m_reservedMemory = std::nullopt;
|
|
|
|
return reservedMemory;
|
|
|
|
}
|
|
|
|
|
2019-04-30 16:32:56 +00:00
|
|
|
void IRGenerationContext::addStateVariable(
|
|
|
|
VariableDeclaration const& _declaration,
|
|
|
|
u256 _storageOffset,
|
|
|
|
unsigned _byteOffset
|
|
|
|
)
|
|
|
|
{
|
|
|
|
m_stateVariables[&_declaration] = make_pair(move(_storageOffset), _byteOffset);
|
|
|
|
}
|
|
|
|
|
2019-03-18 10:21:41 +00:00
|
|
|
string IRGenerationContext::newYulVariable()
|
|
|
|
{
|
|
|
|
return "_" + to_string(++m_varCounter);
|
|
|
|
}
|
|
|
|
|
2020-05-19 19:50:22 +00:00
|
|
|
void IRGenerationContext::initializeInternalDispatch(InternalDispatchMap _internalDispatch)
|
|
|
|
{
|
|
|
|
solAssert(internalDispatchClean(), "");
|
2020-05-14 18:34:59 +00:00
|
|
|
|
2021-03-31 18:03:04 +00:00
|
|
|
for (DispatchSet const& functions: _internalDispatch | ranges::views::values)
|
2020-05-19 19:50:22 +00:00
|
|
|
for (auto function: functions)
|
2020-05-14 18:34:59 +00:00
|
|
|
enqueueFunctionForCodeGeneration(*function);
|
|
|
|
|
2020-05-19 19:50:22 +00:00
|
|
|
m_internalDispatchMap = move(_internalDispatch);
|
|
|
|
}
|
|
|
|
|
|
|
|
InternalDispatchMap IRGenerationContext::consumeInternalDispatchMap()
|
|
|
|
{
|
|
|
|
InternalDispatchMap internalDispatch = move(m_internalDispatchMap);
|
|
|
|
m_internalDispatchMap.clear();
|
|
|
|
return internalDispatch;
|
|
|
|
}
|
|
|
|
|
2020-11-24 12:13:18 +00:00
|
|
|
void IRGenerationContext::addToInternalDispatch(FunctionDefinition const& _function)
|
2020-05-19 19:50:22 +00:00
|
|
|
{
|
2020-11-24 12:13:18 +00:00
|
|
|
FunctionType const* functionType = TypeProvider::function(_function, FunctionType::Kind::Internal);
|
|
|
|
solAssert(functionType, "");
|
2020-05-19 19:50:22 +00:00
|
|
|
|
2020-12-01 15:52:57 +00:00
|
|
|
YulArity arity = YulArity::fromType(*functionType);
|
|
|
|
|
|
|
|
if (m_internalDispatchMap.count(arity) != 0 && m_internalDispatchMap[arity].count(&_function) != 0)
|
|
|
|
// Note that m_internalDispatchMap[arity] is a set with a custom comparator, which looks at function IDs not definitions
|
|
|
|
solAssert(*m_internalDispatchMap[arity].find(&_function) == &_function, "Different definitions with the same function ID");
|
|
|
|
|
|
|
|
m_internalDispatchMap[arity].insert(&_function);
|
2020-11-24 12:13:18 +00:00
|
|
|
enqueueFunctionForCodeGeneration(_function);
|
2020-05-19 19:50:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void IRGenerationContext::internalFunctionCalledThroughDispatch(YulArity const& _arity)
|
|
|
|
{
|
|
|
|
m_internalDispatchMap.try_emplace(_arity);
|
2019-04-02 10:37:48 +00:00
|
|
|
}
|
2019-04-30 16:32:56 +00:00
|
|
|
|
|
|
|
YulUtilFunctions IRGenerationContext::utils()
|
|
|
|
{
|
2020-01-22 14:48:56 +00:00
|
|
|
return YulUtilFunctions(m_evmVersion, m_revertStrings, m_functions);
|
|
|
|
}
|
|
|
|
|
2020-04-09 19:59:17 +00:00
|
|
|
ABIFunctions IRGenerationContext::abiFunctions()
|
|
|
|
{
|
|
|
|
return ABIFunctions(m_evmVersion, m_revertStrings, m_functions);
|
|
|
|
}
|
2021-06-10 03:00:28 +00:00
|
|
|
|
|
|
|
uint64_t IRGenerationContext::internalFunctionID(FunctionDefinition const& _function, bool _requirePresent)
|
|
|
|
{
|
|
|
|
auto [iterator, inserted] = m_functionIDs.try_emplace(_function.id(), m_functionIDs.size() + 1);
|
|
|
|
if (_requirePresent)
|
|
|
|
solAssert(!inserted, "");
|
|
|
|
return iterator->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IRGenerationContext::copyFunctionIDsFrom(IRGenerationContext const& _other)
|
|
|
|
{
|
|
|
|
solAssert(m_functionIDs.empty(), "");
|
|
|
|
m_functionIDs = _other.m_functionIDs;
|
|
|
|
}
|