solidity/libsolidity/codegen/ir/IRGenerator.cpp

1131 lines
38 KiB
C++
Raw Normal View History

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/>.
*/
// SPDX-License-Identifier: GPL-3.0
2019-03-04 22:26:46 +00:00
/**
* @author Alex Beregszaszi
* @date 2017
* Component that translates Solidity code into Yul.
*/
#include <libsolidity/codegen/ir/Common.h>
2019-03-04 22:26:46 +00:00
#include <libsolidity/codegen/ir/IRGenerator.h>
2019-03-18 10:21:41 +00:00
#include <libsolidity/codegen/ir/IRGeneratorForStatements.h>
2019-03-04 22:26:46 +00:00
#include <libsolidity/ast/AST.h>
2019-03-18 10:21:41 +00:00
#include <libsolidity/ast/ASTVisitor.h>
2019-03-04 22:26:46 +00:00
#include <libsolidity/codegen/ABIFunctions.h>
#include <libsolidity/codegen/CompilerUtils.h>
#include <libyul/YulStack.h>
#include <libyul/Utilities.h>
2019-03-04 22:26:46 +00:00
#include <libsolutil/Algorithms.h>
#include <libsolutil/CommonData.h>
#include <libsolutil/StringUtils.h>
#include <libsolutil/Whiskers.h>
2019-03-04 22:26:46 +00:00
#include <liblangutil/SourceReferenceFormatter.h>
#include <sstream>
2020-11-18 17:24:33 +00:00
#include <variant>
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::frontend;
using namespace solidity::langutil;
using namespace solidity::util;
2019-03-04 22:26:46 +00:00
2020-11-18 17:24:33 +00:00
namespace
{
void verifyCallGraph(
set<CallableDeclaration const*, ASTNode::CompareByID> const& _expectedCallables,
set<FunctionDefinition const*> _generatedFunctions
)
{
for (auto const& expectedCallable: _expectedCallables)
if (auto const* expectedFunction = dynamic_cast<FunctionDefinition const*>(expectedCallable))
{
solAssert(
_generatedFunctions.count(expectedFunction) == 1 || expectedFunction->isConstructor(),
"No code generated for function " + expectedFunction->name() + " even though it is not a constructor."
2020-11-18 17:24:33 +00:00
);
_generatedFunctions.erase(expectedFunction);
}
solAssert(
_generatedFunctions.size() == 0,
"Of the generated functions " + toString(_generatedFunctions.size()) + " are not in the call graph."
);
}
set<CallableDeclaration const*, ASTNode::CompareByID> collectReachableCallables(
CallGraph const& _graph
2020-11-18 17:24:33 +00:00
)
{
set<CallableDeclaration const*, ASTNode::CompareByID> reachableCallables;
2021-05-07 11:44:14 +00:00
for (CallGraph::Node const& reachableNode: _graph.edges | ranges::views::keys)
2020-11-18 17:24:33 +00:00
if (holds_alternative<CallableDeclaration const*>(reachableNode))
reachableCallables.emplace(get<CallableDeclaration const*>(reachableNode));
return reachableCallables;
}
}
pair<string, string> IRGenerator::run(
ContractDefinition const& _contract,
2021-06-08 14:35:37 +00:00
bytes const& _cborMetadata,
map<ContractDefinition const*, string_view const> const& _otherYulSources
)
2019-03-04 22:26:46 +00:00
{
string ir = yul::reindent(generate(_contract, _cborMetadata, _otherYulSources));
2019-03-04 22:26:46 +00:00
yul::YulStack asmStack(
m_evmVersion,
m_eofVersion,
yul::YulStack::Language::StrictAssembly,
m_optimiserSettings,
m_context.debugInfoSelection()
);
2019-03-04 22:26:46 +00:00
if (!asmStack.parseAndAnalyze("", ir))
{
string errorMessage;
for (auto const& error: asmStack.errors())
2021-06-29 12:38:59 +00:00
errorMessage += langutil::SourceReferenceFormatter::formatErrorInformation(
*error,
asmStack.charStream("")
);
solAssert(false, ir + "\n\nInvalid IR generated:\n" + errorMessage + "\n");
2019-03-04 22:26:46 +00:00
}
asmStack.optimize();
2022-08-23 17:28:45 +00:00
return {std::move(ir), asmStack.print(m_context.soliditySourceProvider())};
2019-03-04 22:26:46 +00:00
}
string IRGenerator::generate(
ContractDefinition const& _contract,
2021-06-08 14:35:37 +00:00
bytes const& _cborMetadata,
map<ContractDefinition const*, string_view const> const& _otherYulSources
)
2019-03-04 22:26:46 +00:00
{
auto subObjectSources = [&_otherYulSources](std::set<ContractDefinition const*, ASTNode::CompareByID> const& subObjects) -> string
{
std::string subObjectsSources;
for (ContractDefinition const* subObject: subObjects)
subObjectsSources += _otherYulSources.at(subObject);
return subObjectsSources;
};
2021-08-31 10:57:13 +00:00
auto formatUseSrcMap = [](IRGenerationContext const& _context) -> string
{
return joinHumanReadable(
ranges::views::transform(_context.usedSourceNames(), [_context](string const& _sourceName) {
return to_string(_context.sourceIndices().at(_sourceName)) + ":" + escapeAndQuoteString(_sourceName);
}),
", "
);
};
2019-03-04 22:26:46 +00:00
Whiskers t(R"(
/// @use-src <useSrcMapCreation>
2019-03-04 22:26:46 +00:00
object "<CreationObject>" {
code {
2021-08-31 10:57:13 +00:00
<sourceLocationCommentCreation>
2020-07-02 10:48:20 +00:00
<memoryInitCreation>
<callValueCheck>
2020-12-22 16:45:55 +00:00
<?library>
<!library>
<?constructorHasParams> let <constructorParams> := <copyConstructorArguments>() </constructorHasParams>
<constructor>(<constructorParams>)
2020-12-22 16:45:55 +00:00
</library>
2019-03-04 22:26:46 +00:00
<deploy>
<functions>
}
/// @use-src <useSrcMapDeployed>
object "<DeployedObject>" {
2019-03-04 22:26:46 +00:00
code {
2021-08-31 10:57:13 +00:00
<sourceLocationCommentDeployed>
<memoryInitDeployed>
2020-12-22 16:45:55 +00:00
<?library>
let called_via_delegatecall := iszero(eq(loadimmutable("<library_address>"), address()))
</library>
2019-03-04 22:26:46 +00:00
<dispatch>
<deployedFunctions>
2019-03-04 22:26:46 +00:00
}
<deployedSubObjects>
2021-06-08 14:35:37 +00:00
data "<metadataName>" hex"<cborMetadata>"
2019-03-04 22:26:46 +00:00
}
<subObjects>
2019-03-04 22:26:46 +00:00
}
)");
resetContext(_contract, ExecutionContext::Creation);
2020-04-02 18:06:52 +00:00
for (VariableDeclaration const* var: ContractType(_contract).immutableVariables())
m_context.registerImmutableVariable(*var);
t("CreationObject", IRNames::creationObject(_contract));
2021-09-06 16:26:30 +00:00
t("sourceLocationCommentCreation", dispenseLocationComment(_contract));
2020-12-22 16:45:55 +00:00
t("library", _contract.isLibrary());
FunctionDefinition const* constructor = _contract.constructor();
t("callValueCheck", !constructor || !constructor->isPayable() ? callValueCheck() : "");
vector<string> constructorParams;
if (constructor && !constructor->parameters().empty())
{
for (size_t i = 0; i < CompilerUtils::sizeOnStack(constructor->parameters()); ++i)
constructorParams.emplace_back(m_context.newYulVariable());
t(
"copyConstructorArguments",
m_utils.copyConstructorArgumentsToMemoryFunction(_contract, IRNames::creationObject(_contract))
);
}
t("constructorParams", joinHumanReadable(constructorParams));
t("constructorHasParams", !constructorParams.empty());
t("constructor", IRNames::constructor(_contract));
2019-03-04 22:26:46 +00:00
t("deploy", deployCode(_contract));
generateConstructors(_contract);
set<FunctionDefinition const*> creationFunctionList = generateQueuedFunctions();
InternalDispatchMap internalDispatchMap = generateInternalDispatchFunctions(_contract);
2020-11-18 17:24:33 +00:00
t("functions", m_context.functionCollector().requestedFunctions());
t("subObjects", subObjectSources(m_context.subObjectsCreated()));
2020-09-17 15:25:37 +00:00
// This has to be called only after all other code generation for the creation object is complete.
bool creationInvolvesMemoryUnsafeAssembly = m_context.memoryUnsafeInlineAssemblySeen();
t("memoryInitCreation", memoryInit(!creationInvolvesMemoryUnsafeAssembly));
2021-08-31 10:57:13 +00:00
t("useSrcMapCreation", formatUseSrcMap(m_context));
2019-03-04 22:26:46 +00:00
resetContext(_contract, ExecutionContext::Deployed);
// NOTE: Function pointers can be passed from creation code via storage variables. We need to
// get all the functions they could point to into the dispatch functions even if they're never
// referenced by name in the deployed code.
2022-08-23 17:28:45 +00:00
m_context.initializeInternalDispatch(std::move(internalDispatchMap));
2020-04-02 18:06:52 +00:00
// Do not register immutables to avoid assignment.
t("DeployedObject", IRNames::deployedObject(_contract));
2021-09-06 16:26:30 +00:00
t("sourceLocationCommentDeployed", dispenseLocationComment(_contract));
2020-12-22 16:45:55 +00:00
t("library_address", IRNames::libraryAddressImmutable());
2019-03-04 22:26:46 +00:00
t("dispatch", dispatchRoutine(_contract));
set<FunctionDefinition const*> deployedFunctionList = generateQueuedFunctions();
generateInternalDispatchFunctions(_contract);
t("deployedFunctions", m_context.functionCollector().requestedFunctions());
t("deployedSubObjects", subObjectSources(m_context.subObjectsCreated()));
2021-06-08 14:35:37 +00:00
t("metadataName", yul::Object::metadataName());
t("cborMetadata", util::toHex(_cborMetadata));
2021-06-08 14:35:37 +00:00
2021-08-31 10:57:13 +00:00
t("useSrcMapDeployed", formatUseSrcMap(m_context));
2020-09-17 15:25:37 +00:00
// This has to be called only after all other code generation for the deployed object is complete.
bool deployedInvolvesMemoryUnsafeAssembly = m_context.memoryUnsafeInlineAssemblySeen();
t("memoryInitDeployed", memoryInit(!deployedInvolvesMemoryUnsafeAssembly));
2020-11-18 17:24:33 +00:00
solAssert(_contract.annotation().creationCallGraph->get() != nullptr, "");
solAssert(_contract.annotation().deployedCallGraph->get() != nullptr, "");
2022-08-23 17:28:45 +00:00
verifyCallGraph(collectReachableCallables(**_contract.annotation().creationCallGraph), std::move(creationFunctionList));
verifyCallGraph(collectReachableCallables(**_contract.annotation().deployedCallGraph), std::move(deployedFunctionList));
2019-03-04 22:26:46 +00:00
return t.render();
}
2019-03-18 10:21:41 +00:00
string IRGenerator::generate(Block const& _block)
{
IRGeneratorForStatements generator(m_context, m_utils);
generator.generate(_block);
2019-03-18 10:21:41 +00:00
return generator.code();
}
2020-11-18 17:24:33 +00:00
set<FunctionDefinition const*> IRGenerator::generateQueuedFunctions()
{
2020-11-18 17:24:33 +00:00
set<FunctionDefinition const*> functions;
while (!m_context.functionGenerationQueueEmpty())
2020-11-18 17:24:33 +00:00
{
FunctionDefinition const& functionDefinition = *m_context.dequeueFunctionForCodeGeneration();
functions.emplace(&functionDefinition);
// NOTE: generateFunction() may modify function generation queue
2020-11-18 17:24:33 +00:00
generateFunction(functionDefinition);
}
return functions;
}
InternalDispatchMap IRGenerator::generateInternalDispatchFunctions(ContractDefinition const& _contract)
{
solAssert(
m_context.functionGenerationQueueEmpty(),
"At this point all the enqueued functions should have been generated. "
"Otherwise the dispatch may be incomplete."
);
InternalDispatchMap internalDispatchMap = m_context.consumeInternalDispatchMap();
for (YulArity const& arity: internalDispatchMap | ranges::views::keys)
{
string funName = IRNames::internalDispatch(arity);
m_context.functionCollector().createFunction(funName, [&]() {
Whiskers templ(R"(
<sourceLocationComment>
function <functionName>(fun<?+in>, <in></+in>) <?+out>-> <out></+out> {
switch fun
<#cases>
case <funID>
{
<?+out> <out> :=</+out> <name>(<in>)
}
</cases>
2020-09-15 16:57:59 +00:00
default { <panic>() }
}
<sourceLocationComment>
)");
2021-09-06 16:26:30 +00:00
templ("sourceLocationComment", dispenseLocationComment(_contract));
templ("functionName", funName);
2020-10-12 14:01:45 +00:00
templ("panic", m_utils.panicFunction(PanicCode::InvalidInternalFunction));
templ("in", suffixedVariableNameList("in_", 0, arity.in));
templ("out", suffixedVariableNameList("out_", 0, arity.out));
vector<map<string, string>> cases;
for (FunctionDefinition const* function: internalDispatchMap.at(arity))
{
solAssert(function, "");
solAssert(
YulArity::fromType(*TypeProvider::function(*function, FunctionType::Kind::Internal)) == arity,
"A single dispatch function can only handle functions of one arity"
);
solAssert(!function->isConstructor(), "");
// 0 is reserved for uninitialized function pointers
solAssert(function->id() != 0, "Unexpected function ID: 0");
solAssert(m_context.functionCollector().contains(IRNames::function(*function)), "");
cases.emplace_back(map<string, string>{
{"funID", to_string(*function->annotation().internalFunctionID)},
{"name", IRNames::function(*function)}
});
}
2022-08-23 17:28:45 +00:00
templ("cases", std::move(cases));
return templ.render();
});
}
solAssert(m_context.internalDispatchClean(), "");
solAssert(
m_context.functionGenerationQueueEmpty(),
"Internal dispatch generation must not add new functions to generation queue because they won't be proeessed."
);
return internalDispatchMap;
}
2019-03-18 10:21:41 +00:00
string IRGenerator::generateFunction(FunctionDefinition const& _function)
2019-03-04 22:26:46 +00:00
{
string functionName = IRNames::function(_function);
return m_context.functionCollector().createFunction(functionName, [&]() {
2020-11-30 17:59:49 +00:00
m_context.resetLocalVariables();
2019-04-24 21:48:12 +00:00
Whiskers t(R"(
<astIDComment><sourceLocationComment>
2020-05-07 15:29:42 +00:00
function <functionName>(<params>)<?+retParams> -> <retParams></+retParams> {
2020-11-30 17:59:49 +00:00
<retInit>
2019-10-24 17:23:56 +00:00
<body>
2019-04-24 21:48:12 +00:00
}
<contractSourceLocationComment>
2019-04-24 21:48:12 +00:00
)");
if (m_context.debugInfoSelection().astID)
t("astIDComment", "/// @ast-id " + to_string(_function.id()) + "\n");
else
t("astIDComment", "");
2021-09-06 16:26:30 +00:00
t("sourceLocationComment", dispenseLocationComment(_function));
t(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
);
2019-03-04 22:26:46 +00:00
t("functionName", functionName);
2020-05-07 15:29:42 +00:00
vector<string> params;
2019-03-04 22:26:46 +00:00
for (auto const& varDecl: _function.parameters())
2020-05-07 15:29:42 +00:00
params += m_context.addLocalVariable(*varDecl).stackSlots();
t("params", joinHumanReadable(params));
vector<string> retParams;
2020-03-02 20:42:46 +00:00
string retInit;
2019-03-04 22:26:46 +00:00
for (auto const& varDecl: _function.returnParameters())
2020-03-02 20:42:46 +00:00
{
2020-05-07 15:29:42 +00:00
retParams += m_context.addLocalVariable(*varDecl).stackSlots();
2020-03-02 20:42:46 +00:00
retInit += generateInitialAssignment(*varDecl);
}
2020-11-30 17:59:49 +00:00
t("retParams", joinHumanReadable(retParams));
t("retInit", retInit);
if (_function.modifiers().empty())
t("body", generate(_function.body()));
else
{
for (size_t i = 0; i < _function.modifiers().size(); ++i)
{
ModifierInvocation const& modifier = *_function.modifiers().at(i);
string next =
i + 1 < _function.modifiers().size() ?
IRNames::modifierInvocation(*_function.modifiers().at(i + 1)) :
IRNames::functionWithModifierInner(_function);
generateModifier(modifier, _function, next);
}
t("body",
(retParams.empty() ? string{} : joinHumanReadable(retParams) + " := ") +
IRNames::modifierInvocation(*_function.modifiers().at(0)) +
"(" +
joinHumanReadable(retParams + params) +
")"
);
// Now generate the actual inner function.
generateFunctionWithModifierInner(_function);
}
return t.render();
});
}
string IRGenerator::generateModifier(
ModifierInvocation const& _modifierInvocation,
FunctionDefinition const& _function,
string const& _nextFunction
)
{
string functionName = IRNames::modifierInvocation(_modifierInvocation);
return m_context.functionCollector().createFunction(functionName, [&]() {
m_context.resetLocalVariables();
Whiskers t(R"(
<astIDComment><sourceLocationComment>
2020-11-30 17:59:49 +00:00
function <functionName>(<params>)<?+retParams> -> <retParams></+retParams> {
<assignRetParams>
<evalArgs>
<body>
}
<contractSourceLocationComment>
2020-11-30 17:59:49 +00:00
)");
2020-11-30 17:59:49 +00:00
t("functionName", functionName);
vector<string> retParamsIn;
for (auto const& varDecl: _function.returnParameters())
retParamsIn += m_context.addLocalVariable(*varDecl).stackSlots();
2020-11-30 17:59:49 +00:00
vector<string> params = retParamsIn;
for (auto const& varDecl: _function.parameters())
params += m_context.addLocalVariable(*varDecl).stackSlots();
t("params", joinHumanReadable(params));
vector<string> retParams;
string assignRetParams;
for (size_t i = 0; i < retParamsIn.size(); ++i)
{
retParams.emplace_back(m_context.newYulVariable());
assignRetParams += retParams.at(i) + " := " + retParamsIn.at(i) + "\n";
2020-11-30 17:59:49 +00:00
}
t("retParams", joinHumanReadable(retParams));
t("assignRetParams", assignRetParams);
2021-01-13 22:57:03 +00:00
ModifierDefinition const* modifier = dynamic_cast<ModifierDefinition const*>(
_modifierInvocation.name().annotation().referencedDeclaration
);
solAssert(modifier, "");
if (m_context.debugInfoSelection().astID)
t("astIDComment", "/// @ast-id " + to_string(modifier->id()) + "\n");
else
t("astIDComment", "");
2021-09-06 16:26:30 +00:00
t("sourceLocationComment", dispenseLocationComment(*modifier));
t(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
);
2021-01-13 22:57:03 +00:00
switch (*_modifierInvocation.name().annotation().requiredLookup)
{
case VirtualLookup::Virtual:
modifier = &modifier->resolveVirtual(m_context.mostDerivedContract());
solAssert(modifier, "");
break;
case VirtualLookup::Static:
break;
case VirtualLookup::Super:
solAssert(false, "");
}
2020-11-30 17:59:49 +00:00
solAssert(
2021-01-13 22:57:03 +00:00
modifier->parameters().empty() ==
2020-11-30 17:59:49 +00:00
(!_modifierInvocation.arguments() || _modifierInvocation.arguments()->empty()),
""
);
IRGeneratorForStatements expressionEvaluator(m_context, m_utils);
if (_modifierInvocation.arguments())
for (size_t i = 0; i < _modifierInvocation.arguments()->size(); i++)
{
IRVariable argument = expressionEvaluator.evaluateExpression(
*_modifierInvocation.arguments()->at(i),
2021-01-13 22:57:03 +00:00
*modifier->parameters()[i]->annotation().type
2020-11-30 17:59:49 +00:00
);
expressionEvaluator.define(
2021-01-13 22:57:03 +00:00
m_context.addLocalVariable(*modifier->parameters()[i]),
2020-11-30 17:59:49 +00:00
argument
);
}
t("evalArgs", expressionEvaluator.code());
IRGeneratorForStatements generator(m_context, m_utils, [&]() {
string ret = joinHumanReadable(retParams);
return
(ret.empty() ? "" : ret + " := ") +
_nextFunction + "(" + joinHumanReadable(params) + ")\n";
});
2021-01-13 22:57:03 +00:00
generator.generate(modifier->body());
2020-11-30 17:59:49 +00:00
t("body", generator.code());
return t.render();
});
}
string IRGenerator::generateFunctionWithModifierInner(FunctionDefinition const& _function)
{
string functionName = IRNames::functionWithModifierInner(_function);
return m_context.functionCollector().createFunction(functionName, [&]() {
m_context.resetLocalVariables();
Whiskers t(R"(
<sourceLocationComment>
2020-11-30 17:59:49 +00:00
function <functionName>(<params>)<?+retParams> -> <retParams></+retParams> {
<assignRetParams>
<body>
}
<contractSourceLocationComment>
2020-11-30 17:59:49 +00:00
)");
2021-09-06 16:26:30 +00:00
t("sourceLocationComment", dispenseLocationComment(_function));
t(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
);
2020-11-30 17:59:49 +00:00
t("functionName", functionName);
vector<string> retParams;
vector<string> retParamsIn;
for (auto const& varDecl: _function.returnParameters())
retParams += m_context.addLocalVariable(*varDecl).stackSlots();
string assignRetParams;
for (size_t i = 0; i < retParams.size(); ++i)
{
retParamsIn.emplace_back(m_context.newYulVariable());
assignRetParams += retParams.at(i) + " := " + retParamsIn.at(i) + "\n";
2020-11-30 17:59:49 +00:00
}
vector<string> params = retParamsIn;
for (auto const& varDecl: _function.parameters())
params += m_context.addLocalVariable(*varDecl).stackSlots();
t("params", joinHumanReadable(params));
2020-05-07 15:29:42 +00:00
t("retParams", joinHumanReadable(retParams));
2020-11-30 17:59:49 +00:00
t("assignRetParams", assignRetParams);
2019-03-18 10:21:41 +00:00
t("body", generate(_function.body()));
2019-03-04 22:26:46 +00:00
return t.render();
});
}
2019-07-08 20:15:59 +00:00
string IRGenerator::generateGetter(VariableDeclaration const& _varDecl)
{
string functionName = IRNames::function(_varDecl);
2020-05-28 15:39:08 +00:00
return m_context.functionCollector().createFunction(functionName, [&]() {
Type const* type = _varDecl.annotation().type;
2019-07-08 20:15:59 +00:00
2020-05-28 15:39:08 +00:00
solAssert(_varDecl.isStateVariable(), "");
2019-07-08 20:15:59 +00:00
2020-05-28 15:39:08 +00:00
FunctionType accessorType(_varDecl);
TypePointers paramTypes = accessorType.parameterTypes();
if (_varDecl.immutable())
{
solAssert(paramTypes.empty(), "");
solUnimplementedAssert(type->sizeOnStack() == 1);
2020-02-04 13:59:33 +00:00
return Whiskers(R"(
<astIDComment><sourceLocationComment>
2020-05-28 15:39:08 +00:00
function <functionName>() -> rval {
rval := loadimmutable("<id>")
2020-02-04 13:59:33 +00:00
}
<contractSourceLocationComment>
2020-02-04 13:59:33 +00:00
)")
(
"astIDComment",
m_context.debugInfoSelection().astID ?
"/// @ast-id " + to_string(_varDecl.id()) + "\n" :
""
)
2021-09-06 16:26:30 +00:00
("sourceLocationComment", dispenseLocationComment(_varDecl))
(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
)
2020-02-04 13:59:33 +00:00
("functionName", functionName)
2020-05-28 15:39:08 +00:00
("id", to_string(_varDecl.id()))
2020-02-04 13:59:33 +00:00
.render();
2020-05-28 15:39:08 +00:00
}
else if (_varDecl.isConstant())
{
solAssert(paramTypes.empty(), "");
return Whiskers(R"(
<astIDComment><sourceLocationComment>
2020-05-28 15:39:08 +00:00
function <functionName>() -> <ret> {
<ret> := <constantValueFunction>()
}
<contractSourceLocationComment>
2020-05-28 15:39:08 +00:00
)")
(
"astIDComment",
m_context.debugInfoSelection().astID ?
"/// @ast-id " + to_string(_varDecl.id()) + "\n" :
""
)
2021-09-06 16:26:30 +00:00
("sourceLocationComment", dispenseLocationComment(_varDecl))
(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
)
2020-05-28 15:39:08 +00:00
("functionName", functionName)
("constantValueFunction", IRGeneratorForStatements(m_context, m_utils).constantValueFunction(_varDecl))
("ret", suffixedVariableNameList("ret_", 0, _varDecl.type()->sizeOnStack()))
.render();
}
string code;
auto const& location = m_context.storageLocationOfStateVariable(_varDecl);
2020-05-28 15:39:08 +00:00
code += Whiskers(R"(
let slot := <slot>
let offset := <offset>
)")
("slot", location.first.str())
("offset", to_string(location.second))
.render();
if (!paramTypes.empty())
solAssert(
location.second == 0,
"If there are parameters, we are dealing with structs or mappings and thus should have offset zero."
);
// The code of an accessor is of the form `x[a][b][c]` (it is slightly more complicated
// if the final type is a struct).
// In each iteration of the loop below, we consume one parameter, perform an
// index access, reassign the yul variable `slot` and move @a currentType further "down".
// The initial value of @a currentType is only used if we skip the loop completely.
Type const* currentType = _varDecl.annotation().type;
2020-05-28 15:39:08 +00:00
vector<string> parameters;
vector<string> returnVariables;
for (size_t i = 0; i < paramTypes.size(); ++i)
{
MappingType const* mappingType = dynamic_cast<MappingType const*>(currentType);
ArrayType const* arrayType = dynamic_cast<ArrayType const*>(currentType);
solAssert(mappingType || arrayType, "");
vector<string> keys = IRVariable("key_" + to_string(i),
mappingType ? *mappingType->keyType() : *TypeProvider::uint256()
).stackSlots();
parameters += keys;
Whiskers templ(R"(
<?array>
if iszero(lt(<keys>, <length>(slot))) { revert(0, 0) }
</array>
2020-05-28 15:39:08 +00:00
slot<?array>, offset</array> := <indexAccess>(slot<?+keys>, <keys></+keys>)
)");
templ(
2020-05-28 15:39:08 +00:00
"indexAccess",
mappingType ?
m_utils.mappingIndexAccessFunction(*mappingType, *mappingType->keyType()) :
m_utils.storageArrayIndexAccessFunction(*arrayType)
)
("array", arrayType != nullptr)
("keys", joinHumanReadable(keys));
if (arrayType)
templ("length", m_utils.arrayLengthFunction(*arrayType));
code += templ.render();
2020-05-28 15:39:08 +00:00
currentType = mappingType ? mappingType->valueType() : arrayType->baseType();
}
2019-07-08 20:15:59 +00:00
2020-05-28 15:39:08 +00:00
auto returnTypes = accessorType.returnParameterTypes();
solAssert(returnTypes.size() >= 1, "");
if (StructType const* structType = dynamic_cast<StructType const*>(currentType))
{
solAssert(location.second == 0, "");
auto const& names = accessorType.returnParameterNames();
for (size_t i = 0; i < names.size(); ++i)
2020-04-02 18:06:52 +00:00
{
2020-05-28 15:39:08 +00:00
if (returnTypes[i]->category() == Type::Category::Mapping)
continue;
if (
auto const* arrayType = dynamic_cast<ArrayType const*>(returnTypes[i]);
arrayType && !arrayType->isByteArrayOrString()
)
continue;
2020-05-28 15:39:08 +00:00
pair<u256, unsigned> const& offsets = structType->storageOffsetsOfMember(names[i]);
vector<string> retVars = IRVariable("ret_" + to_string(returnVariables.size()), *returnTypes[i]).stackSlots();
returnVariables += retVars;
code += Whiskers(R"(
<ret> := <readStorage>(add(slot, <slotOffset>))
2020-04-02 18:06:52 +00:00
)")
2020-05-28 15:39:08 +00:00
("ret", joinHumanReadable(retVars))
("readStorage", m_utils.readFromStorage(*returnTypes[i], offsets.second, true))
("slotOffset", offsets.first.str())
2020-04-02 18:06:52 +00:00
.render();
}
2020-05-28 15:39:08 +00:00
}
else
{
solAssert(returnTypes.size() == 1, "");
auto const* arrayType = dynamic_cast<ArrayType const*>(returnTypes.front());
if (arrayType)
solAssert(arrayType->isByteArrayOrString(), "");
2020-05-28 15:39:08 +00:00
vector<string> retVars = IRVariable("ret", *returnTypes.front()).stackSlots();
returnVariables += retVars;
code += Whiskers(R"(
<ret> := <readStorage>(slot, offset)
)")
("ret", joinHumanReadable(retVars))
("readStorage", m_utils.readFromStorageDynamic(*returnTypes.front(), true))
.render();
}
2020-04-02 18:06:52 +00:00
2020-05-28 15:39:08 +00:00
return Whiskers(R"(
<astIDComment><sourceLocationComment>
2020-05-28 15:39:08 +00:00
function <functionName>(<params>) -> <retVariables> {
<code>
2020-04-02 18:06:52 +00:00
}
<contractSourceLocationComment>
2020-05-28 15:39:08 +00:00
)")
("functionName", functionName)
("params", joinHumanReadable(parameters))
("retVariables", joinHumanReadable(returnVariables))
("code", std::move(code))
(
"astIDComment",
m_context.debugInfoSelection().astID ?
"/// @ast-id " + to_string(_varDecl.id()) + "\n" :
""
)
2021-09-06 16:26:30 +00:00
("sourceLocationComment", dispenseLocationComment(_varDecl))
(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
)
2020-05-28 15:39:08 +00:00
.render();
});
2019-07-08 20:15:59 +00:00
}
string IRGenerator::generateExternalFunction(ContractDefinition const& _contract, FunctionType const& _functionType)
{
string functionName = IRNames::externalFunctionABIWrapper(_functionType.declaration());
return m_context.functionCollector().createFunction(functionName, [&](vector<string>&, vector<string>&) -> string {
Whiskers t(R"X(
<callValueCheck>
<?+params>let <params> := </+params> <abiDecode>(4, calldatasize())
<?+retParams>let <retParams> := </+retParams> <function>(<params>)
let memPos := <allocateUnbounded>()
let memEnd := <abiEncode>(memPos <?+retParams>,</+retParams> <retParams>)
return(memPos, sub(memEnd, memPos))
)X");
t("callValueCheck", (_functionType.isPayable() || _contract.isLibrary()) ? "" : callValueCheck());
unsigned paramVars = make_shared<TupleType>(_functionType.parameterTypes())->sizeOnStack();
unsigned retVars = make_shared<TupleType>(_functionType.returnParameterTypes())->sizeOnStack();
ABIFunctions abiFunctions(m_evmVersion, m_context.revertStrings(), m_context.functionCollector());
t("abiDecode", abiFunctions.tupleDecoder(_functionType.parameterTypes()));
t("params", suffixedVariableNameList("param_", 0, paramVars));
t("retParams", suffixedVariableNameList("ret_", 0, retVars));
if (FunctionDefinition const* funDef = dynamic_cast<FunctionDefinition const*>(&_functionType.declaration()))
{
solAssert(!funDef->isConstructor());
t("function", m_context.enqueueFunctionForCodeGeneration(*funDef));
}
else if (VariableDeclaration const* varDecl = dynamic_cast<VariableDeclaration const*>(&_functionType.declaration()))
t("function", generateGetter(*varDecl));
else
solAssert(false, "Unexpected declaration for function!");
t("allocateUnbounded", m_utils.allocateUnboundedFunction());
t("abiEncode", abiFunctions.tupleEncoder(_functionType.returnParameterTypes(), _functionType.returnParameterTypes(), _contract.isLibrary()));
return t.render();
});
}
2020-03-02 20:42:46 +00:00
string IRGenerator::generateInitialAssignment(VariableDeclaration const& _varDecl)
{
IRGeneratorForStatements generator(m_context, m_utils);
generator.initializeLocalVar(_varDecl);
return generator.code();
}
2020-05-07 15:29:42 +00:00
pair<string, map<ContractDefinition const*, vector<string>>> IRGenerator::evaluateConstructorArguments(
ContractDefinition const& _contract
)
2019-03-04 22:26:46 +00:00
{
struct InheritanceOrder
{
bool operator()(ContractDefinition const* _c1, ContractDefinition const* _c2) const
{
solAssert(util::contains(linearizedBaseContracts, _c1) && util::contains(linearizedBaseContracts, _c2), "");
auto it1 = find(linearizedBaseContracts.begin(), linearizedBaseContracts.end(), _c1);
auto it2 = find(linearizedBaseContracts.begin(), linearizedBaseContracts.end(), _c2);
return it1 < it2;
}
vector<ContractDefinition const*> const& linearizedBaseContracts;
} inheritanceOrder{_contract.annotation().linearizedBaseContracts};
2020-05-07 15:29:42 +00:00
map<ContractDefinition const*, vector<string>> constructorParams;
map<ContractDefinition const*, std::vector<ASTPointer<Expression>>const *, InheritanceOrder>
baseConstructorArguments(inheritanceOrder);
for (ASTPointer<InheritanceSpecifier> const& base: _contract.baseContracts())
if (FunctionDefinition const* baseConstructor = dynamic_cast<ContractDefinition const*>(
base->name().annotation().referencedDeclaration
2020-05-07 15:29:42 +00:00
)->constructor(); baseConstructor && base->arguments())
solAssert(baseConstructorArguments.emplace(
dynamic_cast<ContractDefinition const*>(baseConstructor->scope()),
base->arguments()
).second, "");
if (FunctionDefinition const* constructor = _contract.constructor())
for (ASTPointer<ModifierInvocation> const& modifier: constructor->modifiers())
if (auto const* baseContract = dynamic_cast<ContractDefinition const*>(
modifier->name().annotation().referencedDeclaration
))
if (
FunctionDefinition const* baseConstructor = baseContract->constructor();
baseConstructor && modifier->arguments()
)
solAssert(baseConstructorArguments.emplace(
dynamic_cast<ContractDefinition const*>(baseConstructor->scope()),
modifier->arguments()
).second, "");
IRGeneratorForStatements generator{m_context, m_utils};
for (auto&& [baseContract, arguments]: baseConstructorArguments)
{
solAssert(baseContract && arguments, "");
if (baseContract->constructor() && !arguments->empty())
{
vector<string> params;
for (size_t i = 0; i < arguments->size(); ++i)
2020-05-07 15:29:42 +00:00
params += generator.evaluateExpression(
*(arguments->at(i)),
*(baseContract->constructor()->parameters()[i]->type())
).stackSlots();
constructorParams[baseContract] = std::move(params);
}
}
return {generator.code(), constructorParams};
}
string IRGenerator::initStateVariables(ContractDefinition const& _contract)
{
IRGeneratorForStatements generator{m_context, m_utils};
for (VariableDeclaration const* variable: _contract.stateVariables())
2020-04-02 18:06:52 +00:00
if (!variable->isConstant())
generator.initializeStateVar(*variable);
return generator.code();
}
void IRGenerator::generateConstructors(ContractDefinition const& _contract)
{
auto listAllParams =
[&](map<ContractDefinition const*, vector<string>> const& baseParams) -> vector<string>
{
vector<string> params;
for (ContractDefinition const* contract: _contract.annotation().linearizedBaseContracts)
if (baseParams.count(contract))
params += baseParams.at(contract);
return params;
};
2020-05-07 15:29:42 +00:00
map<ContractDefinition const*, vector<string>> baseConstructorParams;
for (size_t i = 0; i < _contract.annotation().linearizedBaseContracts.size(); ++i)
{
ContractDefinition const* contract = _contract.annotation().linearizedBaseContracts[i];
baseConstructorParams.erase(contract);
2020-12-02 17:00:04 +00:00
m_context.resetLocalVariables();
m_context.functionCollector().createFunction(IRNames::constructor(*contract), [&]() {
Whiskers t(R"(
2021-09-09 16:24:28 +00:00
<astIDComment><sourceLocationComment>
function <functionName>(<params><comma><baseParams>) {
<evalBaseArguments>
<sourceLocationComment>
<?hasNextConstructor> <nextConstructor>(<nextParams>) </hasNextConstructor>
<initStateVariables>
<userDefinedConstructorBody>
}
<contractSourceLocationComment>
)");
2020-05-07 15:29:42 +00:00
vector<string> params;
if (contract->constructor())
for (ASTPointer<VariableDeclaration> const& varDecl: contract->constructor()->parameters())
2020-05-07 15:29:42 +00:00
params += m_context.addLocalVariable(*varDecl).stackSlots();
if (m_context.debugInfoSelection().astID && contract->constructor())
2021-09-09 16:24:28 +00:00
t("astIDComment", "/// @ast-id " + to_string(contract->constructor()->id()) + "\n");
else
t("astIDComment", "");
2021-09-06 16:26:30 +00:00
t("sourceLocationComment", dispenseLocationComment(
contract->constructor() ?
2021-09-06 16:18:33 +00:00
dynamic_cast<ASTNode const&>(*contract->constructor()) :
dynamic_cast<ASTNode const&>(*contract)
));
t(
"contractSourceLocationComment",
2021-09-06 16:26:30 +00:00
dispenseLocationComment(m_context.mostDerivedContract())
);
2020-05-07 15:29:42 +00:00
t("params", joinHumanReadable(params));
vector<string> baseParams = listAllParams(baseConstructorParams);
t("baseParams", joinHumanReadable(baseParams));
t("comma", !params.empty() && !baseParams.empty() ? ", " : "");
t("functionName", IRNames::constructor(*contract));
2020-05-07 15:29:42 +00:00
pair<string, map<ContractDefinition const*, vector<string>>> evaluatedArgs = evaluateConstructorArguments(*contract);
baseConstructorParams.insert(evaluatedArgs.second.begin(), evaluatedArgs.second.end());
t("evalBaseArguments", evaluatedArgs.first);
if (i < _contract.annotation().linearizedBaseContracts.size() - 1)
{
t("hasNextConstructor", true);
ContractDefinition const* nextContract = _contract.annotation().linearizedBaseContracts[i + 1];
t("nextConstructor", IRNames::constructor(*nextContract));
2020-05-07 15:29:42 +00:00
t("nextParams", joinHumanReadable(listAllParams(baseConstructorParams)));
}
else
t("hasNextConstructor", false);
t("initStateVariables", initStateVariables(*contract));
2020-12-02 17:00:04 +00:00
string body;
if (FunctionDefinition const* constructor = contract->constructor())
{
vector<ModifierInvocation*> realModifiers;
for (auto const& modifierInvocation: constructor->modifiers())
// Filter out the base constructor calls
if (dynamic_cast<ModifierDefinition const*>(modifierInvocation->name().annotation().referencedDeclaration))
realModifiers.emplace_back(modifierInvocation.get());
if (realModifiers.empty())
body = generate(constructor->body());
else
{
for (size_t i = 0; i < realModifiers.size(); ++i)
{
ModifierInvocation const& modifier = *realModifiers.at(i);
string next =
i + 1 < realModifiers.size() ?
IRNames::modifierInvocation(*realModifiers.at(i + 1)) :
IRNames::functionWithModifierInner(*constructor);
generateModifier(modifier, *constructor, next);
}
body =
IRNames::modifierInvocation(*realModifiers.at(0)) +
2020-12-02 17:00:04 +00:00
"(" +
joinHumanReadable(params) +
")";
// Now generate the actual inner function.
generateFunctionWithModifierInner(*constructor);
}
}
2022-08-23 17:28:45 +00:00
t("userDefinedConstructorBody", std::move(body));
return t.render();
});
}
2019-03-04 22:26:46 +00:00
}
string IRGenerator::deployCode(ContractDefinition const& _contract)
{
Whiskers t(R"X(
2021-03-15 15:57:56 +00:00
let <codeOffset> := <allocateUnbounded>()
codecopy(<codeOffset>, dataoffset("<object>"), datasize("<object>"))
<#immutables>
setimmutable(<codeOffset>, "<immutableName>", <value>)
</immutables>
return(<codeOffset>, datasize("<object>"))
2019-03-04 22:26:46 +00:00
)X");
2021-03-15 15:57:56 +00:00
t("allocateUnbounded", m_utils.allocateUnboundedFunction());
t("codeOffset", m_context.newYulVariable());
t("object", IRNames::deployedObject(_contract));
2020-04-02 18:06:52 +00:00
2021-03-15 15:57:56 +00:00
vector<map<string, string>> immutables;
2020-12-22 16:45:55 +00:00
if (_contract.isLibrary())
2020-04-02 18:06:52 +00:00
{
2020-12-22 16:45:55 +00:00
solAssert(ContractType(_contract).immutableVariables().empty(), "");
2021-03-15 15:57:56 +00:00
immutables.emplace_back(map<string, string>{
{"immutableName"s, IRNames::libraryAddressImmutable()},
{"value"s, "address()"}
2020-04-02 18:06:52 +00:00
});
2020-12-22 16:45:55 +00:00
2020-04-02 18:06:52 +00:00
}
2020-12-22 16:45:55 +00:00
else
for (VariableDeclaration const* immutable: ContractType(_contract).immutableVariables())
{
solUnimplementedAssert(immutable->type()->isValueType());
solUnimplementedAssert(immutable->type()->sizeOnStack() == 1);
2021-03-15 15:57:56 +00:00
immutables.emplace_back(map<string, string>{
{"immutableName"s, to_string(immutable->id())},
{"value"s, "mload(" + to_string(m_context.immutableMemoryOffset(*immutable)) + ")"}
2020-12-22 16:45:55 +00:00
});
}
2021-03-15 15:57:56 +00:00
t("immutables", std::move(immutables));
2019-03-04 22:26:46 +00:00
return t.render();
}
string IRGenerator::callValueCheck()
{
2021-03-15 15:45:00 +00:00
return "if callvalue() { " + m_utils.revertReasonIfDebugFunction("Ether sent to non-payable function") + "() }";
2019-03-04 22:26:46 +00:00
}
string IRGenerator::dispatchRoutine(ContractDefinition const& _contract)
{
Whiskers t(R"X(
<?+cases>if iszero(lt(calldatasize(), 4))
2019-03-04 22:26:46 +00:00
{
let selector := <shr224>(calldataload(0))
switch selector
<#cases>
case <functionSelector>
{
// <functionName>
2020-12-22 16:45:55 +00:00
<delegatecallCheck>
<externalFunction>()
2019-03-04 22:26:46 +00:00
}
</cases>
default {}
}</+cases>
<?+receiveEther>if iszero(calldatasize()) { <receiveEther> }</+receiveEther>
2019-03-04 22:26:46 +00:00
<fallback>
)X");
t("shr224", m_utils.shiftRightFunction(224));
vector<map<string, string>> functions;
for (auto const& function: _contract.interfaceFunctions())
{
2020-04-01 02:39:38 +00:00
functions.emplace_back();
2019-03-04 22:26:46 +00:00
map<string, string>& templ = functions.back();
templ["functionSelector"] = "0x" + function.first.hex();
FunctionTypePointer const& type = function.second;
templ["functionName"] = type->externalSignature();
2020-12-22 16:45:55 +00:00
string delegatecallCheck;
if (_contract.isLibrary())
{
solAssert(!type->isPayable(), "");
if (type->stateMutability() > StateMutability::View)
// If the function is not a view function and is called without DELEGATECALL,
// we revert.
2020-11-27 20:26:13 +00:00
delegatecallCheck =
"if iszero(called_via_delegatecall) { " +
2021-03-15 15:45:00 +00:00
m_utils.revertReasonIfDebugFunction("Non-view function of library called without DELEGATECALL") +
"() }";
2020-12-22 16:45:55 +00:00
}
templ["delegatecallCheck"] = delegatecallCheck;
2019-07-08 20:15:59 +00:00
templ["externalFunction"] = generateExternalFunction(_contract, *type);
2019-03-04 22:26:46 +00:00
}
t("cases", functions);
2020-11-27 20:26:13 +00:00
FunctionDefinition const* etherReceiver = _contract.receiveFunction();
if (etherReceiver)
2020-12-22 16:45:55 +00:00
{
solAssert(!_contract.isLibrary(), "");
t("receiveEther", m_context.enqueueFunctionForCodeGeneration(*etherReceiver) + "() stop()");
}
else
t("receiveEther", "");
2019-03-04 22:26:46 +00:00
if (FunctionDefinition const* fallback = _contract.fallbackFunction())
{
2020-12-22 16:45:55 +00:00
solAssert(!_contract.isLibrary(), "");
2019-03-04 22:26:46 +00:00
string fallbackCode;
if (!fallback->isPayable())
fallbackCode += callValueCheck() + "\n";
if (fallback->parameters().empty())
fallbackCode += m_context.enqueueFunctionForCodeGeneration(*fallback) + "() stop()";
else
{
solAssert(fallback->parameters().size() == 1 && fallback->returnParameters().size() == 1, "");
fallbackCode += "let retval := " + m_context.enqueueFunctionForCodeGeneration(*fallback) + "(0, calldatasize())\n";
fallbackCode += "return(add(retval, 0x20), mload(retval))\n";
}
2019-03-04 22:26:46 +00:00
t("fallback", fallbackCode);
}
else
2021-03-15 15:45:00 +00:00
t("fallback", (
2020-11-27 20:26:13 +00:00
etherReceiver ?
2021-03-15 15:45:00 +00:00
m_utils.revertReasonIfDebugFunction("Unknown signature and no fallback defined") :
m_utils.revertReasonIfDebugFunction("Contract does not have fallback nor receive functions")
) + "()");
2019-03-04 22:26:46 +00:00
return t.render();
}
2020-07-02 10:48:20 +00:00
string IRGenerator::memoryInit(bool _useMemoryGuard)
2019-03-04 22:26:46 +00:00
{
// This function should be called at the beginning of the EVM call frame
// and thus can assume all memory to be zero, including the contents of
// the "zero memory area" (the position CompilerUtils::zeroPointer points to).
return
2020-07-02 10:48:20 +00:00
Whiskers{
_useMemoryGuard ?
"mstore(<memPtr>, memoryguard(<freeMemoryStart>))" :
"mstore(<memPtr>, <freeMemoryStart>)"
}
2019-03-04 22:26:46 +00:00
("memPtr", to_string(CompilerUtils::freeMemoryPointer))
2020-07-02 10:48:20 +00:00
(
"freeMemoryStart",
to_string(CompilerUtils::generalPurposeMemoryStart + m_context.reservedMemory())
).render();
2019-03-04 22:26:46 +00:00
}
void IRGenerator::resetContext(ContractDefinition const& _contract, ExecutionContext _context)
2019-03-04 22:26:46 +00:00
{
solAssert(
m_context.functionGenerationQueueEmpty(),
"Reset function generation queue while it still had functions."
);
2019-03-04 22:26:46 +00:00
solAssert(
m_context.functionCollector().requestedFunctions().empty(),
2019-03-04 22:26:46 +00:00
"Reset context while it still had functions."
);
solAssert(
m_context.internalDispatchClean(),
"Reset internal dispatch map without consuming it."
);
IRGenerationContext newContext(
m_evmVersion,
_context,
m_context.revertStrings(),
m_optimiserSettings,
m_context.sourceIndices(),
m_context.debugInfoSelection(),
m_context.soliditySourceProvider()
);
2022-08-23 17:28:45 +00:00
m_context = std::move(newContext);
m_context.setMostDerivedContract(_contract);
for (auto const& var: ContractType(_contract).stateVariables())
m_context.addStateVariable(*get<0>(var), get<1>(var), get<2>(var));
2019-03-04 22:26:46 +00:00
}
2021-09-06 16:18:33 +00:00
2021-09-06 16:26:30 +00:00
string IRGenerator::dispenseLocationComment(ASTNode const& _node)
2021-09-06 16:18:33 +00:00
{
2021-09-06 16:26:30 +00:00
return ::dispenseLocationComment(_node, m_context);
2021-09-06 16:18:33 +00:00
}