mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Refactoring yul source locations.
This commit is contained in:
parent
fe4822a1d2
commit
e404b6e7a6
@ -482,7 +482,7 @@ void ControlFlowBuilder::operator()(yul::Identifier const& _identifier)
|
|||||||
m_currentNode->variableOccurrences.emplace_back(
|
m_currentNode->variableOccurrences.emplace_back(
|
||||||
*declaration,
|
*declaration,
|
||||||
VariableOccurrence::Kind::Access,
|
VariableOccurrence::Kind::Access,
|
||||||
_identifier.location
|
_identifier.debugData->location
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -498,7 +498,7 @@ void ControlFlowBuilder::operator()(yul::Assignment const& _assignment)
|
|||||||
m_currentNode->variableOccurrences.emplace_back(
|
m_currentNode->variableOccurrences.emplace_back(
|
||||||
*declaration,
|
*declaration,
|
||||||
VariableOccurrence::Kind::Assignment,
|
VariableOccurrence::Kind::Assignment,
|
||||||
variable.location
|
variable.debugData->location
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -201,9 +201,9 @@ bool ReferencesResolver::visit(Return const& _return)
|
|||||||
|
|
||||||
void ReferencesResolver::operator()(yul::FunctionDefinition const& _function)
|
void ReferencesResolver::operator()(yul::FunctionDefinition const& _function)
|
||||||
{
|
{
|
||||||
validateYulIdentifierName(_function.name, _function.location);
|
validateYulIdentifierName(_function.name, _function.debugData->location);
|
||||||
for (yul::TypedName const& varName: _function.parameters + _function.returnVariables)
|
for (yul::TypedName const& varName: _function.parameters + _function.returnVariables)
|
||||||
validateYulIdentifierName(varName.name, varName.location);
|
validateYulIdentifierName(varName.name, varName.debugData->location);
|
||||||
|
|
||||||
bool wasInsideFunction = m_yulInsideFunction;
|
bool wasInsideFunction = m_yulInsideFunction;
|
||||||
m_yulInsideFunction = true;
|
m_yulInsideFunction = true;
|
||||||
@ -238,7 +238,7 @@ void ReferencesResolver::operator()(yul::Identifier const& _identifier)
|
|||||||
{
|
{
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
4718_error,
|
4718_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Multiple matching identifiers. Resolving overloaded identifiers is not supported."
|
"Multiple matching identifiers. Resolving overloaded identifiers is not supported."
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
@ -251,7 +251,7 @@ void ReferencesResolver::operator()(yul::Identifier const& _identifier)
|
|||||||
)
|
)
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
9467_error,
|
9467_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Identifier not found. Use \".slot\" and \".offset\" to access storage variables."
|
"Identifier not found. Use \".slot\" and \".offset\" to access storage variables."
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
@ -261,7 +261,7 @@ void ReferencesResolver::operator()(yul::Identifier const& _identifier)
|
|||||||
{
|
{
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
6578_error,
|
6578_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Cannot access local Solidity variables from inside an inline assembly function."
|
"Cannot access local Solidity variables from inside an inline assembly function."
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
@ -275,7 +275,7 @@ void ReferencesResolver::operator()(yul::VariableDeclaration const& _varDecl)
|
|||||||
{
|
{
|
||||||
for (auto const& identifier: _varDecl.variables)
|
for (auto const& identifier: _varDecl.variables)
|
||||||
{
|
{
|
||||||
validateYulIdentifierName(identifier.name, identifier.location);
|
validateYulIdentifierName(identifier.name, identifier.debugData->location);
|
||||||
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
@ -289,7 +289,7 @@ void ReferencesResolver::operator()(yul::VariableDeclaration const& _varDecl)
|
|||||||
if (!ssl.infos.empty())
|
if (!ssl.infos.empty())
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
3859_error,
|
3859_error,
|
||||||
identifier.location,
|
identifier.debugData->location,
|
||||||
ssl,
|
ssl,
|
||||||
"This declaration shadows a declaration outside the inline assembly block."
|
"This declaration shadows a declaration outside the inline assembly block."
|
||||||
);
|
);
|
||||||
|
@ -743,7 +743,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
solAssert(var->type(), "Expected variable type!");
|
solAssert(var->type(), "Expected variable type!");
|
||||||
if (var->immutable())
|
if (var->immutable())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(3773_error, _identifier.location, "Assembly access to immutable variables is not supported.");
|
m_errorReporter.typeError(3773_error, _identifier.debugData->location, "Assembly access to immutable variables is not supported.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (var->isConstant())
|
if (var->isConstant())
|
||||||
@ -752,7 +752,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
3558_error,
|
3558_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Constant variable is circular."
|
"Constant variable is circular."
|
||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
@ -762,24 +762,24 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
|
|
||||||
if (var && !var->value())
|
if (var && !var->value())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(3224_error, _identifier.location, "Constant has no value.");
|
m_errorReporter.typeError(3224_error, _identifier.debugData->location, "Constant has no value.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (_context == yul::IdentifierContext::LValue)
|
else if (_context == yul::IdentifierContext::LValue)
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(6252_error, _identifier.location, "Constant variables cannot be assigned to.");
|
m_errorReporter.typeError(6252_error, _identifier.debugData->location, "Constant variables cannot be assigned to.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (!identifierInfo.suffix.empty())
|
else if (!identifierInfo.suffix.empty())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(6617_error, _identifier.location, "The suffixes .offset and .slot can only be used on non-constant storage variables.");
|
m_errorReporter.typeError(6617_error, _identifier.debugData->location, "The suffixes .offset and .slot can only be used on non-constant storage variables.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (var && var->value() && !var->value()->annotation().type && !dynamic_cast<Literal const*>(var->value().get()))
|
else if (var && var->value() && !var->value()->annotation().type && !dynamic_cast<Literal const*>(var->value().get()))
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
2249_error,
|
2249_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Constant variables with non-literal values cannot be forward referenced from inline assembly."
|
"Constant variables with non-literal values cannot be forward referenced from inline assembly."
|
||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
@ -789,7 +789,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
type(*var->value())->category() != Type::Category::RationalNumber
|
type(*var->value())->category() != Type::Category::RationalNumber
|
||||||
))
|
))
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(7615_error, _identifier.location, "Only direct number constants and references to such constants are supported by inline assembly.");
|
m_errorReporter.typeError(7615_error, _identifier.debugData->location, "Only direct number constants and references to such constants are supported by inline assembly.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -804,19 +804,19 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
if (suffix != "slot" && suffix != "offset")
|
if (suffix != "slot" && suffix != "offset")
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(4656_error, _identifier.location, "State variables only support \".slot\" and \".offset\".");
|
m_errorReporter.typeError(4656_error, _identifier.debugData->location, "State variables only support \".slot\" and \".offset\".");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (_context == yul::IdentifierContext::LValue)
|
else if (_context == yul::IdentifierContext::LValue)
|
||||||
{
|
{
|
||||||
if (var->isStateVariable())
|
if (var->isStateVariable())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(4713_error, _identifier.location, "State variables cannot be assigned to - you have to use \"sstore()\".");
|
m_errorReporter.typeError(4713_error, _identifier.debugData->location, "State variables cannot be assigned to - you have to use \"sstore()\".");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (suffix != "slot")
|
else if (suffix != "slot")
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(9739_error, _identifier.location, "Only .slot can be assigned to.");
|
m_errorReporter.typeError(9739_error, _identifier.debugData->location, "Only .slot can be assigned to.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -828,13 +828,13 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
if (suffix != "offset" && suffix != "length")
|
if (suffix != "offset" && suffix != "length")
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(1536_error, _identifier.location, "Calldata variables only support \".offset\" and \".length\".");
|
m_errorReporter.typeError(1536_error, _identifier.debugData->location, "Calldata variables only support \".offset\" and \".length\".");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(3622_error, _identifier.location, "The suffix \"." + suffix + "\" is not supported by this variable or type.");
|
m_errorReporter.typeError(3622_error, _identifier.debugData->location, "The suffix \"." + suffix + "\" is not supported by this variable or type.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -842,14 +842,14 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
1408_error,
|
1408_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Only local variables are supported. To access storage variables, use the \".slot\" and \".offset\" suffixes."
|
"Only local variables are supported. To access storage variables, use the \".slot\" and \".offset\" suffixes."
|
||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (var->type()->dataStoredIn(DataLocation::Storage))
|
else if (var->type()->dataStoredIn(DataLocation::Storage))
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(9068_error, _identifier.location, "You have to use the \".slot\" or \".offset\" suffix to access storage reference variables.");
|
m_errorReporter.typeError(9068_error, _identifier.debugData->location, "You have to use the \".slot\" or \".offset\" suffix to access storage reference variables.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (var->type()->sizeOnStack() != 1)
|
else if (var->type()->sizeOnStack() != 1)
|
||||||
@ -858,18 +858,18 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
auto const* arrayType = dynamic_cast<ArrayType const*>(var->type());
|
auto const* arrayType = dynamic_cast<ArrayType const*>(var->type());
|
||||||
arrayType && arrayType->isDynamicallySized() && arrayType->dataStoredIn(DataLocation::CallData)
|
arrayType && arrayType->isDynamicallySized() && arrayType->dataStoredIn(DataLocation::CallData)
|
||||||
)
|
)
|
||||||
m_errorReporter.typeError(1397_error, _identifier.location, "Call data elements cannot be accessed directly. Use \".offset\" and \".length\" to access the calldata offset and length of this array and then use \"calldatacopy\".");
|
m_errorReporter.typeError(1397_error, _identifier.debugData->location, "Call data elements cannot be accessed directly. Use \".offset\" and \".length\" to access the calldata offset and length of this array and then use \"calldatacopy\".");
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(!var->type()->dataStoredIn(DataLocation::CallData), "");
|
solAssert(!var->type()->dataStoredIn(DataLocation::CallData), "");
|
||||||
m_errorReporter.typeError(9857_error, _identifier.location, "Only types that use one stack slot are supported.");
|
m_errorReporter.typeError(9857_error, _identifier.debugData->location, "Only types that use one stack slot are supported.");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (!identifierInfo.suffix.empty())
|
else if (!identifierInfo.suffix.empty())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(7944_error, _identifier.location, "The suffixes \".offset\", \".slot\" and \".length\" can only be used with variables.");
|
m_errorReporter.typeError(7944_error, _identifier.debugData->location, "The suffixes \".offset\", \".slot\" and \".length\" can only be used with variables.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (_context == yul::IdentifierContext::LValue)
|
else if (_context == yul::IdentifierContext::LValue)
|
||||||
@ -877,7 +877,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
if (dynamic_cast<MagicVariableDeclaration const*>(declaration))
|
if (dynamic_cast<MagicVariableDeclaration const*>(declaration))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
m_errorReporter.typeError(1990_error, _identifier.location, "Only local variables can be assigned to in inline assembly.");
|
m_errorReporter.typeError(1990_error, _identifier.debugData->location, "Only local variables can be assigned to in inline assembly.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -886,7 +886,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
solAssert(!!declaration->type(), "Type of declaration required but not yet determined.");
|
solAssert(!!declaration->type(), "Type of declaration required but not yet determined.");
|
||||||
if (dynamic_cast<FunctionDefinition const*>(declaration))
|
if (dynamic_cast<FunctionDefinition const*>(declaration))
|
||||||
{
|
{
|
||||||
m_errorReporter.declarationError(2025_error, _identifier.location, "Access to functions is not allowed in inline assembly.");
|
m_errorReporter.declarationError(2025_error, _identifier.debugData->location, "Access to functions is not allowed in inline assembly.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (dynamic_cast<VariableDeclaration const*>(declaration))
|
else if (dynamic_cast<VariableDeclaration const*>(declaration))
|
||||||
@ -896,7 +896,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
if (!contract->isLibrary())
|
if (!contract->isLibrary())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(4977_error, _identifier.location, "Expected a library.");
|
m_errorReporter.typeError(4977_error, _identifier.debugData->location, "Expected a library.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -69,7 +69,7 @@ public:
|
|||||||
if (yul::EVMDialect const* dialect = dynamic_cast<decltype(dialect)>(&m_dialect))
|
if (yul::EVMDialect const* dialect = dynamic_cast<decltype(dialect)>(&m_dialect))
|
||||||
if (yul::BuiltinFunctionForEVM const* fun = dialect->builtin(_funCall.functionName.name))
|
if (yul::BuiltinFunctionForEVM const* fun = dialect->builtin(_funCall.functionName.name))
|
||||||
if (fun->instruction)
|
if (fun->instruction)
|
||||||
checkInstruction(_funCall.location, *fun->instruction);
|
checkInstruction(_funCall.debugData->location, *fun->instruction);
|
||||||
|
|
||||||
for (auto const& arg: _funCall.arguments)
|
for (auto const& arg: _funCall.arguments)
|
||||||
std::visit(*this, arg);
|
std::visit(*this, arg);
|
||||||
|
@ -175,7 +175,7 @@ void ASTJsonConverter::appendExpressionAttributes(
|
|||||||
Json::Value ASTJsonConverter::inlineAssemblyIdentifierToJson(pair<yul::Identifier const* ,InlineAssemblyAnnotation::ExternalIdentifierInfo> _info) const
|
Json::Value ASTJsonConverter::inlineAssemblyIdentifierToJson(pair<yul::Identifier const* ,InlineAssemblyAnnotation::ExternalIdentifierInfo> _info) const
|
||||||
{
|
{
|
||||||
Json::Value tuple(Json::objectValue);
|
Json::Value tuple(Json::objectValue);
|
||||||
tuple["src"] = sourceLocationToString(_info.first->location);
|
tuple["src"] = sourceLocationToString(_info.first->debugData->location);
|
||||||
tuple["declaration"] = idOrNull(_info.second.declaration);
|
tuple["declaration"] = idOrNull(_info.second.declaration);
|
||||||
tuple["isSlot"] = Json::Value(_info.second.suffix == "slot");
|
tuple["isSlot"] = Json::Value(_info.second.suffix == "slot");
|
||||||
tuple["isOffset"] = Json::Value(_info.second.suffix == "offset");
|
tuple["isOffset"] = Json::Value(_info.second.suffix == "offset");
|
||||||
|
@ -24,12 +24,10 @@
|
|||||||
#include <libsolidity/ast/ASTJsonImporter.h>
|
#include <libsolidity/ast/ASTJsonImporter.h>
|
||||||
|
|
||||||
#include <libyul/AsmJsonImporter.h>
|
#include <libyul/AsmJsonImporter.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AST.h>
|
#include <libyul/AST.h>
|
||||||
#include <libyul/Dialect.h>
|
#include <libyul/Dialect.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/SourceLocation.h>
|
#include <liblangutil/SourceLocation.h>
|
||||||
|
@ -28,11 +28,11 @@
|
|||||||
#include <libsolidity/codegen/CompilerUtils.h>
|
#include <libsolidity/codegen/CompilerUtils.h>
|
||||||
#include <libsolidity/interface/Version.h>
|
#include <libsolidity/interface/Version.h>
|
||||||
|
|
||||||
|
#include <libyul/AST.h>
|
||||||
#include <libyul/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AST.h>
|
|
||||||
#include <libyul/backends/evm/AsmCodeGen.h>
|
#include <libyul/backends/evm/AsmCodeGen.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/backends/evm/EVMMetrics.h>
|
#include <libyul/backends/evm/EVMMetrics.h>
|
||||||
@ -426,7 +426,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
if (stackDiff < 1 || stackDiff > 16)
|
if (stackDiff < 1 || stackDiff > 16)
|
||||||
BOOST_THROW_EXCEPTION(
|
BOOST_THROW_EXCEPTION(
|
||||||
StackTooDeepError() <<
|
StackTooDeepError() <<
|
||||||
errinfo_sourceLocation(_identifier.location) <<
|
errinfo_sourceLocation(_identifier.debugData->location) <<
|
||||||
util::errinfo_comment("Stack too deep (" + to_string(stackDiff) + "), try removing local variables.")
|
util::errinfo_comment("Stack too deep (" + to_string(stackDiff) + "), try removing local variables.")
|
||||||
);
|
);
|
||||||
if (_context == yul::IdentifierContext::RValue)
|
if (_context == yul::IdentifierContext::RValue)
|
||||||
|
@ -192,9 +192,9 @@ private:
|
|||||||
solAssert(false, "");
|
solAssert(false, "");
|
||||||
|
|
||||||
if (isdigit(value.front()))
|
if (isdigit(value.front()))
|
||||||
return yul::Literal{_identifier.location, yul::LiteralKind::Number, yul::YulString{value}, {}};
|
return yul::Literal{_identifier.debugData, yul::LiteralKind::Number, yul::YulString{value}, {}};
|
||||||
else
|
else
|
||||||
return yul::Identifier{_identifier.location, yul::YulString{value}};
|
return yul::Identifier{_identifier.debugData, yul::YulString{value}};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -61,8 +61,8 @@
|
|||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libyul/AsmJsonConverter.h>
|
#include <libyul/AsmJsonConverter.h>
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AST.h>
|
#include <libyul/AST.h>
|
||||||
|
#include <libyul/AsmParser.h>
|
||||||
|
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/SemVerHandler.h>
|
#include <liblangutil/SemVerHandler.h>
|
||||||
|
@ -24,8 +24,8 @@
|
|||||||
#include <libsolidity/parsing/Parser.h>
|
#include <libsolidity/parsing/Parser.h>
|
||||||
|
|
||||||
#include <libsolidity/interface/Version.h>
|
#include <libsolidity/interface/Version.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AST.h>
|
#include <libyul/AST.h>
|
||||||
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
@ -1299,7 +1299,7 @@ ASTPointer<InlineAssembly> Parser::parseInlineAssembly(ASTPointer<ASTString> con
|
|||||||
if (block == nullptr)
|
if (block == nullptr)
|
||||||
BOOST_THROW_EXCEPTION(FatalError());
|
BOOST_THROW_EXCEPTION(FatalError());
|
||||||
|
|
||||||
location.end = block->location.end;
|
location.end = block->debugData->location.end;
|
||||||
return make_shared<InlineAssembly>(nextID(), location, _docString, dialect, block);
|
return make_shared<InlineAssembly>(nextID(), location, _docString, dialect, block);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
60
libyul/AST.h
60
libyul/AST.h
@ -35,56 +35,80 @@ namespace solidity::yul
|
|||||||
|
|
||||||
using Type = YulString;
|
using Type = YulString;
|
||||||
|
|
||||||
struct TypedName { langutil::SourceLocation location; YulString name; Type type; };
|
struct DebugData
|
||||||
|
{
|
||||||
|
explicit DebugData(langutil::SourceLocation _location): location(std::move(_location)) {}
|
||||||
|
langutil::SourceLocation location;
|
||||||
|
static std::shared_ptr<DebugData const> create(langutil::SourceLocation _location = {})
|
||||||
|
{
|
||||||
|
return std::make_shared<DebugData const>(_location);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct TypedName { std::shared_ptr<DebugData const> debugData; YulString name; Type type; };
|
||||||
using TypedNameList = std::vector<TypedName>;
|
using TypedNameList = std::vector<TypedName>;
|
||||||
|
|
||||||
/// Literal number or string (up to 32 bytes)
|
/// Literal number or string (up to 32 bytes)
|
||||||
enum class LiteralKind { Number, Boolean, String };
|
enum class LiteralKind { Number, Boolean, String };
|
||||||
struct Literal { langutil::SourceLocation location; LiteralKind kind; YulString value; Type type; };
|
struct Literal { std::shared_ptr<DebugData const> debugData; LiteralKind kind; YulString value; Type type; };
|
||||||
/// External / internal identifier or label reference
|
/// External / internal identifier or label reference
|
||||||
struct Identifier { langutil::SourceLocation location; YulString name; };
|
struct Identifier { std::shared_ptr<DebugData const> debugData; YulString name; };
|
||||||
/// Assignment ("x := mload(20:u256)", expects push-1-expression on the right hand
|
/// Assignment ("x := mload(20:u256)", expects push-1-expression on the right hand
|
||||||
/// side and requires x to occupy exactly one stack slot.
|
/// side and requires x to occupy exactly one stack slot.
|
||||||
///
|
///
|
||||||
/// Multiple assignment ("x, y := f()"), where the left hand side variables each occupy
|
/// Multiple assignment ("x, y := f()"), where the left hand side variables each occupy
|
||||||
/// a single stack slot and expects a single expression on the right hand returning
|
/// a single stack slot and expects a single expression on the right hand returning
|
||||||
/// the same amount of items as the number of variables.
|
/// the same amount of items as the number of variables.
|
||||||
struct Assignment { langutil::SourceLocation location; std::vector<Identifier> variableNames; std::unique_ptr<Expression> value; };
|
struct Assignment { std::shared_ptr<DebugData const> debugData; std::vector<Identifier> variableNames; std::unique_ptr<Expression> value; };
|
||||||
struct FunctionCall { langutil::SourceLocation location; Identifier functionName; std::vector<Expression> arguments; };
|
struct FunctionCall { std::shared_ptr<DebugData const> debugData; Identifier functionName; std::vector<Expression> arguments; };
|
||||||
/// Statement that contains only a single expression
|
/// Statement that contains only a single expression
|
||||||
struct ExpressionStatement { langutil::SourceLocation location; Expression expression; };
|
struct ExpressionStatement { std::shared_ptr<DebugData const> debugData; Expression expression; };
|
||||||
/// Block-scope variable declaration ("let x:u256 := mload(20:u256)"), non-hoisted
|
/// Block-scope variable declaration ("let x:u256 := mload(20:u256)"), non-hoisted
|
||||||
struct VariableDeclaration { langutil::SourceLocation location; TypedNameList variables; std::unique_ptr<Expression> value; };
|
struct VariableDeclaration { std::shared_ptr<DebugData const> debugData; TypedNameList variables; std::unique_ptr<Expression> value; };
|
||||||
/// Block that creates a scope (frees declared stack variables)
|
/// Block that creates a scope (frees declared stack variables)
|
||||||
struct Block { langutil::SourceLocation location; std::vector<Statement> statements; };
|
struct Block { std::shared_ptr<DebugData const> debugData; std::vector<Statement> statements; };
|
||||||
/// Function definition ("function f(a, b) -> (d, e) { ... }")
|
/// Function definition ("function f(a, b) -> (d, e) { ... }")
|
||||||
struct FunctionDefinition { langutil::SourceLocation location; YulString name; TypedNameList parameters; TypedNameList returnVariables; Block body; };
|
struct FunctionDefinition { std::shared_ptr<DebugData const> debugData; YulString name; TypedNameList parameters; TypedNameList returnVariables; Block body; };
|
||||||
/// Conditional execution without "else" part.
|
/// Conditional execution without "else" part.
|
||||||
struct If { langutil::SourceLocation location; std::unique_ptr<Expression> condition; Block body; };
|
struct If { std::shared_ptr<DebugData const> debugData; std::unique_ptr<Expression> condition; Block body; };
|
||||||
/// Switch case or default case
|
/// Switch case or default case
|
||||||
struct Case { langutil::SourceLocation location; std::unique_ptr<Literal> value; Block body; };
|
struct Case { std::shared_ptr<DebugData const> debugData; std::unique_ptr<Literal> value; Block body; };
|
||||||
/// Switch statement
|
/// Switch statement
|
||||||
struct Switch { langutil::SourceLocation location; std::unique_ptr<Expression> expression; std::vector<Case> cases; };
|
struct Switch { std::shared_ptr<DebugData const> debugData; std::unique_ptr<Expression> expression; std::vector<Case> cases; };
|
||||||
struct ForLoop { langutil::SourceLocation location; Block pre; std::unique_ptr<Expression> condition; Block post; Block body; };
|
struct ForLoop { std::shared_ptr<DebugData const> debugData; Block pre; std::unique_ptr<Expression> condition; Block post; Block body; };
|
||||||
/// Break statement (valid within for loop)
|
/// Break statement (valid within for loop)
|
||||||
struct Break { langutil::SourceLocation location; };
|
struct Break { std::shared_ptr<DebugData const> debugData; };
|
||||||
/// Continue statement (valid within for loop)
|
/// Continue statement (valid within for loop)
|
||||||
struct Continue { langutil::SourceLocation location; };
|
struct Continue { std::shared_ptr<DebugData const> debugData; };
|
||||||
/// Leave statement (valid within function)
|
/// Leave statement (valid within function)
|
||||||
struct Leave { langutil::SourceLocation location; };
|
struct Leave { std::shared_ptr<DebugData const> debugData; };
|
||||||
|
|
||||||
struct LocationExtractor
|
struct LocationExtractor
|
||||||
{
|
{
|
||||||
template <class T> langutil::SourceLocation operator()(T const& _node) const
|
template <class T> langutil::SourceLocation operator()(T const& _node) const
|
||||||
{
|
{
|
||||||
return _node.location;
|
return _node.debugData ? _node.debugData->location : langutil::SourceLocation{};
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Extracts the source location from an inline assembly node.
|
/// Extracts the source location from a Yul node.
|
||||||
template <class T> inline langutil::SourceLocation locationOf(T const& _node)
|
template <class T> inline langutil::SourceLocation locationOf(T const& _node)
|
||||||
{
|
{
|
||||||
return std::visit(LocationExtractor(), _node);
|
return std::visit(LocationExtractor(), _node);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct DebugDataExtractor
|
||||||
|
{
|
||||||
|
template <class T> std::shared_ptr<DebugData const> const& operator()(T const& _node) const
|
||||||
|
{
|
||||||
|
return _node.debugData;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Extracts the debug data from a Yul node.
|
||||||
|
template <class T> inline std::shared_ptr<DebugData const> const& debugDataOf(T const& _node)
|
||||||
|
{
|
||||||
|
return std::visit(DebugDataExtractor(), _node);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
namespace solidity::yul
|
namespace solidity::yul
|
||||||
{
|
{
|
||||||
|
|
||||||
|
struct DebugData;
|
||||||
enum class LiteralKind;
|
enum class LiteralKind;
|
||||||
struct Literal;
|
struct Literal;
|
||||||
struct Label;
|
struct Label;
|
||||||
|
@ -98,22 +98,22 @@ AsmAnalysisInfo AsmAnalyzer::analyzeStrictAssertCorrect(Dialect const& _dialect,
|
|||||||
|
|
||||||
vector<YulString> AsmAnalyzer::operator()(Literal const& _literal)
|
vector<YulString> AsmAnalyzer::operator()(Literal const& _literal)
|
||||||
{
|
{
|
||||||
expectValidType(_literal.type, _literal.location);
|
expectValidType(_literal.type, _literal.debugData->location);
|
||||||
if (_literal.kind == LiteralKind::String && _literal.value.str().size() > 32)
|
if (_literal.kind == LiteralKind::String && _literal.value.str().size() > 32)
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
3069_error,
|
3069_error,
|
||||||
_literal.location,
|
_literal.debugData->location,
|
||||||
"String literal too long (" + to_string(_literal.value.str().size()) + " > 32)"
|
"String literal too long (" + to_string(_literal.value.str().size()) + " > 32)"
|
||||||
);
|
);
|
||||||
else if (_literal.kind == LiteralKind::Number && bigint(_literal.value.str()) > u256(-1))
|
else if (_literal.kind == LiteralKind::Number && bigint(_literal.value.str()) > u256(-1))
|
||||||
m_errorReporter.typeError(6708_error, _literal.location, "Number literal too large (> 256 bits)");
|
m_errorReporter.typeError(6708_error, _literal.debugData->location, "Number literal too large (> 256 bits)");
|
||||||
else if (_literal.kind == LiteralKind::Boolean)
|
else if (_literal.kind == LiteralKind::Boolean)
|
||||||
yulAssert(_literal.value == "true"_yulstring || _literal.value == "false"_yulstring, "");
|
yulAssert(_literal.value == "true"_yulstring || _literal.value == "false"_yulstring, "");
|
||||||
|
|
||||||
if (!m_dialect.validTypeForLiteral(_literal.kind, _literal.value, _literal.type))
|
if (!m_dialect.validTypeForLiteral(_literal.kind, _literal.value, _literal.type))
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
5170_error,
|
5170_error,
|
||||||
_literal.location,
|
_literal.debugData->location,
|
||||||
"Invalid type \"" + _literal.type.str() + "\" for literal \"" + _literal.value.str() + "\"."
|
"Invalid type \"" + _literal.type.str() + "\" for literal \"" + _literal.value.str() + "\"."
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -133,7 +133,7 @@ vector<YulString> AsmAnalyzer::operator()(Identifier const& _identifier)
|
|||||||
if (!m_activeVariables.count(&_var))
|
if (!m_activeVariables.count(&_var))
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
4990_error,
|
4990_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Variable " + _identifier.name.str() + " used before it was declared."
|
"Variable " + _identifier.name.str() + " used before it was declared."
|
||||||
);
|
);
|
||||||
type = _var.type;
|
type = _var.type;
|
||||||
@ -142,7 +142,7 @@ vector<YulString> AsmAnalyzer::operator()(Identifier const& _identifier)
|
|||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
6041_error,
|
6041_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Function " + _identifier.name.str() + " used without being called."
|
"Function " + _identifier.name.str() + " used without being called."
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -160,7 +160,7 @@ vector<YulString> AsmAnalyzer::operator()(Identifier const& _identifier)
|
|||||||
// Only add an error message if the callback did not do it.
|
// Only add an error message if the callback did not do it.
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
8198_error,
|
8198_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Identifier \"" + _identifier.name.str() + "\" not found."
|
"Identifier \"" + _identifier.name.str() + "\" not found."
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -176,7 +176,7 @@ void AsmAnalyzer::operator()(ExpressionStatement const& _statement)
|
|||||||
if (watcher.ok() && !types.empty())
|
if (watcher.ok() && !types.empty())
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
3083_error,
|
3083_error,
|
||||||
_statement.location,
|
_statement.debugData->location,
|
||||||
"Top-level expressions are not supposed to return values (this expression returns " +
|
"Top-level expressions are not supposed to return values (this expression returns " +
|
||||||
to_string(types.size()) +
|
to_string(types.size()) +
|
||||||
" value" +
|
" value" +
|
||||||
@ -196,7 +196,7 @@ void AsmAnalyzer::operator()(Assignment const& _assignment)
|
|||||||
if (!variables.insert(_variableName.name).second)
|
if (!variables.insert(_variableName.name).second)
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
9005_error,
|
9005_error,
|
||||||
_assignment.location,
|
_assignment.debugData->location,
|
||||||
"Variable " +
|
"Variable " +
|
||||||
_variableName.name.str() +
|
_variableName.name.str() +
|
||||||
" occurs multiple times on the left-hand side of the assignment."
|
" occurs multiple times on the left-hand side of the assignment."
|
||||||
@ -207,7 +207,7 @@ void AsmAnalyzer::operator()(Assignment const& _assignment)
|
|||||||
if (types.size() != numVariables)
|
if (types.size() != numVariables)
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
8678_error,
|
8678_error,
|
||||||
_assignment.location,
|
_assignment.debugData->location,
|
||||||
"Variable count for assignment to \"" +
|
"Variable count for assignment to \"" +
|
||||||
joinHumanReadable(applyMap(_assignment.variableNames, [](auto const& _identifier){ return _identifier.name.str(); })) +
|
joinHumanReadable(applyMap(_assignment.variableNames, [](auto const& _identifier){ return _identifier.name.str(); })) +
|
||||||
"\" does not match number of values (" +
|
"\" does not match number of values (" +
|
||||||
@ -229,14 +229,14 @@ void AsmAnalyzer::operator()(VariableDeclaration const& _varDecl)
|
|||||||
for (auto const& variable: _varDecl.variables)
|
for (auto const& variable: _varDecl.variables)
|
||||||
// Call the resolver for variable declarations to allow it to raise errors on shadowing.
|
// Call the resolver for variable declarations to allow it to raise errors on shadowing.
|
||||||
m_resolver(
|
m_resolver(
|
||||||
yul::Identifier{variable.location, variable.name},
|
yul::Identifier{variable.debugData, variable.name},
|
||||||
yul::IdentifierContext::VariableDeclaration,
|
yul::IdentifierContext::VariableDeclaration,
|
||||||
m_currentScope->insideFunction()
|
m_currentScope->insideFunction()
|
||||||
);
|
);
|
||||||
for (auto const& variable: _varDecl.variables)
|
for (auto const& variable: _varDecl.variables)
|
||||||
{
|
{
|
||||||
expectValidIdentifier(variable.name, variable.location);
|
expectValidIdentifier(variable.name, variable.debugData->location);
|
||||||
expectValidType(variable.type, variable.location);
|
expectValidType(variable.type, variable.debugData->location);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_varDecl.value)
|
if (_varDecl.value)
|
||||||
@ -245,7 +245,7 @@ void AsmAnalyzer::operator()(VariableDeclaration const& _varDecl)
|
|||||||
if (types.size() != numVariables)
|
if (types.size() != numVariables)
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
3812_error,
|
3812_error,
|
||||||
_varDecl.location,
|
_varDecl.debugData->location,
|
||||||
"Variable count mismatch for declaration of \"" +
|
"Variable count mismatch for declaration of \"" +
|
||||||
joinHumanReadable(applyMap(_varDecl.variables, [](auto const& _identifier){ return _identifier.name.str(); })) +
|
joinHumanReadable(applyMap(_varDecl.variables, [](auto const& _identifier){ return _identifier.name.str(); })) +
|
||||||
+ "\": " +
|
+ "\": " +
|
||||||
@ -264,7 +264,7 @@ void AsmAnalyzer::operator()(VariableDeclaration const& _varDecl)
|
|||||||
if (variable.type != givenType)
|
if (variable.type != givenType)
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
3947_error,
|
3947_error,
|
||||||
variable.location,
|
variable.debugData->location,
|
||||||
"Assigning value of type \"" + givenType.str() + "\" to variable of type \"" + variable.type.str() + "\"."
|
"Assigning value of type \"" + givenType.str() + "\" to variable of type \"" + variable.type.str() + "\"."
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -279,14 +279,14 @@ void AsmAnalyzer::operator()(VariableDeclaration const& _varDecl)
|
|||||||
void AsmAnalyzer::operator()(FunctionDefinition const& _funDef)
|
void AsmAnalyzer::operator()(FunctionDefinition const& _funDef)
|
||||||
{
|
{
|
||||||
yulAssert(!_funDef.name.empty(), "");
|
yulAssert(!_funDef.name.empty(), "");
|
||||||
expectValidIdentifier(_funDef.name, _funDef.location);
|
expectValidIdentifier(_funDef.name, _funDef.debugData->location);
|
||||||
Block const* virtualBlock = m_info.virtualBlocks.at(&_funDef).get();
|
Block const* virtualBlock = m_info.virtualBlocks.at(&_funDef).get();
|
||||||
yulAssert(virtualBlock, "");
|
yulAssert(virtualBlock, "");
|
||||||
Scope& varScope = scope(virtualBlock);
|
Scope& varScope = scope(virtualBlock);
|
||||||
for (auto const& var: _funDef.parameters + _funDef.returnVariables)
|
for (auto const& var: _funDef.parameters + _funDef.returnVariables)
|
||||||
{
|
{
|
||||||
expectValidIdentifier(var.name, var.location);
|
expectValidIdentifier(var.name, var.debugData->location);
|
||||||
expectValidType(var.type, var.location);
|
expectValidType(var.type, var.debugData->location);
|
||||||
m_activeVariables.insert(&std::get<Scope::Variable>(varScope.identifiers.at(var.name)));
|
m_activeVariables.insert(&std::get<Scope::Variable>(varScope.identifiers.at(var.name)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -315,7 +315,7 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
4202_error,
|
4202_error,
|
||||||
_funCall.functionName.location,
|
_funCall.functionName.debugData->location,
|
||||||
"Attempt to call variable instead of function."
|
"Attempt to call variable instead of function."
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -329,7 +329,7 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
if (!validateInstructions(_funCall))
|
if (!validateInstructions(_funCall))
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
4619_error,
|
4619_error,
|
||||||
_funCall.functionName.location,
|
_funCall.functionName.debugData->location,
|
||||||
"Function \"" + _funCall.functionName.name.str() + "\" not found."
|
"Function \"" + _funCall.functionName.name.str() + "\" not found."
|
||||||
);
|
);
|
||||||
yulAssert(!watcher.ok(), "Expected a reported error.");
|
yulAssert(!watcher.ok(), "Expected a reported error.");
|
||||||
@ -338,7 +338,7 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
if (parameterTypes && _funCall.arguments.size() != parameterTypes->size())
|
if (parameterTypes && _funCall.arguments.size() != parameterTypes->size())
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
7000_error,
|
7000_error,
|
||||||
_funCall.functionName.location,
|
_funCall.functionName.debugData->location,
|
||||||
"Function \"" + _funCall.functionName.name.str() + "\" expects " +
|
"Function \"" + _funCall.functionName.name.str() + "\" expects " +
|
||||||
to_string(parameterTypes->size()) +
|
to_string(parameterTypes->size()) +
|
||||||
" arguments but got " +
|
" arguments but got " +
|
||||||
@ -358,13 +358,13 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
if (!holds_alternative<Literal>(arg))
|
if (!holds_alternative<Literal>(arg))
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
9114_error,
|
9114_error,
|
||||||
_funCall.functionName.location,
|
_funCall.functionName.debugData->location,
|
||||||
"Function expects direct literals as arguments."
|
"Function expects direct literals as arguments."
|
||||||
);
|
);
|
||||||
else if (*literalArgumentKind != get<Literal>(arg).kind)
|
else if (*literalArgumentKind != get<Literal>(arg).kind)
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
5859_error,
|
5859_error,
|
||||||
get<Literal>(arg).location,
|
get<Literal>(arg).debugData->location,
|
||||||
"Function expects " + to_string(*literalArgumentKind) + " literal."
|
"Function expects " + to_string(*literalArgumentKind) + " literal."
|
||||||
);
|
);
|
||||||
else if (*literalArgumentKind == LiteralKind::String)
|
else if (*literalArgumentKind == LiteralKind::String)
|
||||||
@ -375,7 +375,7 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
if (!m_dataNames.count(get<Literal>(arg).value))
|
if (!m_dataNames.count(get<Literal>(arg).value))
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
3517_error,
|
3517_error,
|
||||||
get<Literal>(arg).location,
|
get<Literal>(arg).debugData->location,
|
||||||
"Unknown data object \"" + std::get<Literal>(arg).value.str() + "\"."
|
"Unknown data object \"" + std::get<Literal>(arg).value.str() + "\"."
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -384,7 +384,7 @@ vector<YulString> AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
|||||||
if (get<Literal>(arg).value.empty())
|
if (get<Literal>(arg).value.empty())
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
1844_error,
|
1844_error,
|
||||||
get<Literal>(arg).location,
|
get<Literal>(arg).debugData->location,
|
||||||
"The \"verbatim_*\" builtins cannot be used with empty bytecode."
|
"The \"verbatim_*\" builtins cannot be used with empty bytecode."
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -427,7 +427,7 @@ void AsmAnalyzer::operator()(Switch const& _switch)
|
|||||||
if (_switch.cases.size() == 1 && !_switch.cases[0].value)
|
if (_switch.cases.size() == 1 && !_switch.cases[0].value)
|
||||||
m_errorReporter.warning(
|
m_errorReporter.warning(
|
||||||
9592_error,
|
9592_error,
|
||||||
_switch.location,
|
_switch.debugData->location,
|
||||||
"\"switch\" statement with only a default case."
|
"\"switch\" statement with only a default case."
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -440,7 +440,7 @@ void AsmAnalyzer::operator()(Switch const& _switch)
|
|||||||
{
|
{
|
||||||
auto watcher = m_errorReporter.errorWatcher();
|
auto watcher = m_errorReporter.errorWatcher();
|
||||||
|
|
||||||
expectType(valueType, _case.value->type, _case.value->location);
|
expectType(valueType, _case.value->type, _case.value->debugData->location);
|
||||||
|
|
||||||
// We cannot use "expectExpression" here because *_case.value is not an
|
// We cannot use "expectExpression" here because *_case.value is not an
|
||||||
// Expression and would be converted to an Expression otherwise.
|
// Expression and would be converted to an Expression otherwise.
|
||||||
@ -450,7 +450,7 @@ void AsmAnalyzer::operator()(Switch const& _switch)
|
|||||||
if (watcher.ok() && !cases.insert(valueOfLiteral(*_case.value)).second)
|
if (watcher.ok() && !cases.insert(valueOfLiteral(*_case.value)).second)
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
6792_error,
|
6792_error,
|
||||||
_case.location,
|
_case.debugData->location,
|
||||||
"Duplicate case \"" +
|
"Duplicate case \"" +
|
||||||
valueOfLiteral(*_case.value).str() +
|
valueOfLiteral(*_case.value).str() +
|
||||||
"\" defined."
|
"\" defined."
|
||||||
@ -542,11 +542,11 @@ void AsmAnalyzer::checkAssignment(Identifier const& _variable, YulString _valueT
|
|||||||
if (Scope::Identifier const* var = m_currentScope->lookup(_variable.name))
|
if (Scope::Identifier const* var = m_currentScope->lookup(_variable.name))
|
||||||
{
|
{
|
||||||
if (!holds_alternative<Scope::Variable>(*var))
|
if (!holds_alternative<Scope::Variable>(*var))
|
||||||
m_errorReporter.typeError(2657_error, _variable.location, "Assignment requires variable.");
|
m_errorReporter.typeError(2657_error, _variable.debugData->location, "Assignment requires variable.");
|
||||||
else if (!m_activeVariables.count(&std::get<Scope::Variable>(*var)))
|
else if (!m_activeVariables.count(&std::get<Scope::Variable>(*var)))
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
1133_error,
|
1133_error,
|
||||||
_variable.location,
|
_variable.debugData->location,
|
||||||
"Variable " + _variable.name.str() + " used before it was declared."
|
"Variable " + _variable.name.str() + " used before it was declared."
|
||||||
);
|
);
|
||||||
else
|
else
|
||||||
@ -565,11 +565,11 @@ void AsmAnalyzer::checkAssignment(Identifier const& _variable, YulString _valueT
|
|||||||
|
|
||||||
if (!found && watcher.ok())
|
if (!found && watcher.ok())
|
||||||
// Only add message if the callback did not.
|
// Only add message if the callback did not.
|
||||||
m_errorReporter.declarationError(4634_error, _variable.location, "Variable not found or variable not lvalue.");
|
m_errorReporter.declarationError(4634_error, _variable.debugData->location, "Variable not found or variable not lvalue.");
|
||||||
if (variableType && *variableType != _valueType)
|
if (variableType && *variableType != _valueType)
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
9547_error,
|
9547_error,
|
||||||
_variable.location,
|
_variable.debugData->location,
|
||||||
"Assigning a value of type \"" +
|
"Assigning a value of type \"" +
|
||||||
_valueType.str() +
|
_valueType.str() +
|
||||||
"\" to a variable of type \"" +
|
"\" to a variable of type \"" +
|
||||||
@ -713,5 +713,5 @@ bool AsmAnalyzer::validateInstructions(evmasm::Instruction _instr, SourceLocatio
|
|||||||
|
|
||||||
bool AsmAnalyzer::validateInstructions(FunctionCall const& _functionCall)
|
bool AsmAnalyzer::validateInstructions(FunctionCall const& _functionCall)
|
||||||
{
|
{
|
||||||
return validateInstructions(_functionCall.functionName.name.str(), _functionCall.functionName.location);
|
return validateInstructions(_functionCall.functionName.name.str(), _functionCall.functionName.debugData->location);
|
||||||
}
|
}
|
||||||
|
@ -32,7 +32,7 @@ namespace solidity::yul
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Block const& _node) const
|
Json::Value AsmJsonConverter::operator()(Block const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulBlock");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulBlock");
|
||||||
ret["statements"] = vectorOfVariantsToJson(_node.statements);
|
ret["statements"] = vectorOfVariantsToJson(_node.statements);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -40,7 +40,7 @@ Json::Value AsmJsonConverter::operator()(Block const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(TypedName const& _node) const
|
Json::Value AsmJsonConverter::operator()(TypedName const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(!_node.name.empty(), "Invalid variable name.");
|
yulAssert(!_node.name.empty(), "Invalid variable name.");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulTypedName");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulTypedName");
|
||||||
ret["name"] = _node.name.str();
|
ret["name"] = _node.name.str();
|
||||||
ret["type"] = _node.type.str();
|
ret["type"] = _node.type.str();
|
||||||
return ret;
|
return ret;
|
||||||
@ -48,7 +48,7 @@ Json::Value AsmJsonConverter::operator()(TypedName const& _node) const
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Literal const& _node) const
|
Json::Value AsmJsonConverter::operator()(Literal const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulLiteral");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulLiteral");
|
||||||
switch (_node.kind)
|
switch (_node.kind)
|
||||||
{
|
{
|
||||||
case LiteralKind::Number:
|
case LiteralKind::Number:
|
||||||
@ -73,7 +73,7 @@ Json::Value AsmJsonConverter::operator()(Literal const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(Identifier const& _node) const
|
Json::Value AsmJsonConverter::operator()(Identifier const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(!_node.name.empty(), "Invalid identifier");
|
yulAssert(!_node.name.empty(), "Invalid identifier");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulIdentifier");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulIdentifier");
|
||||||
ret["name"] = _node.name.str();
|
ret["name"] = _node.name.str();
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -81,7 +81,7 @@ Json::Value AsmJsonConverter::operator()(Identifier const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(Assignment const& _node) const
|
Json::Value AsmJsonConverter::operator()(Assignment const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(_node.variableNames.size() >= 1, "Invalid assignment syntax");
|
yulAssert(_node.variableNames.size() >= 1, "Invalid assignment syntax");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulAssignment");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulAssignment");
|
||||||
for (auto const& var: _node.variableNames)
|
for (auto const& var: _node.variableNames)
|
||||||
ret["variableNames"].append((*this)(var));
|
ret["variableNames"].append((*this)(var));
|
||||||
ret["value"] = _node.value ? std::visit(*this, *_node.value) : Json::nullValue;
|
ret["value"] = _node.value ? std::visit(*this, *_node.value) : Json::nullValue;
|
||||||
@ -90,7 +90,7 @@ Json::Value AsmJsonConverter::operator()(Assignment const& _node) const
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(FunctionCall const& _node) const
|
Json::Value AsmJsonConverter::operator()(FunctionCall const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulFunctionCall");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulFunctionCall");
|
||||||
ret["functionName"] = (*this)(_node.functionName);
|
ret["functionName"] = (*this)(_node.functionName);
|
||||||
ret["arguments"] = vectorOfVariantsToJson(_node.arguments);
|
ret["arguments"] = vectorOfVariantsToJson(_node.arguments);
|
||||||
return ret;
|
return ret;
|
||||||
@ -98,14 +98,14 @@ Json::Value AsmJsonConverter::operator()(FunctionCall const& _node) const
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(ExpressionStatement const& _node) const
|
Json::Value AsmJsonConverter::operator()(ExpressionStatement const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulExpressionStatement");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulExpressionStatement");
|
||||||
ret["expression"] = std::visit(*this, _node.expression);
|
ret["expression"] = std::visit(*this, _node.expression);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(VariableDeclaration const& _node) const
|
Json::Value AsmJsonConverter::operator()(VariableDeclaration const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulVariableDeclaration");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulVariableDeclaration");
|
||||||
for (auto const& var: _node.variables)
|
for (auto const& var: _node.variables)
|
||||||
ret["variables"].append((*this)(var));
|
ret["variables"].append((*this)(var));
|
||||||
|
|
||||||
@ -117,7 +117,7 @@ Json::Value AsmJsonConverter::operator()(VariableDeclaration const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(FunctionDefinition const& _node) const
|
Json::Value AsmJsonConverter::operator()(FunctionDefinition const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(!_node.name.empty(), "Invalid function name.");
|
yulAssert(!_node.name.empty(), "Invalid function name.");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulFunctionDefinition");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulFunctionDefinition");
|
||||||
ret["name"] = _node.name.str();
|
ret["name"] = _node.name.str();
|
||||||
for (auto const& var: _node.parameters)
|
for (auto const& var: _node.parameters)
|
||||||
ret["parameters"].append((*this)(var));
|
ret["parameters"].append((*this)(var));
|
||||||
@ -130,7 +130,7 @@ Json::Value AsmJsonConverter::operator()(FunctionDefinition const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(If const& _node) const
|
Json::Value AsmJsonConverter::operator()(If const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(_node.condition, "Invalid if condition.");
|
yulAssert(_node.condition, "Invalid if condition.");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulIf");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulIf");
|
||||||
ret["condition"] = std::visit(*this, *_node.condition);
|
ret["condition"] = std::visit(*this, *_node.condition);
|
||||||
ret["body"] = (*this)(_node.body);
|
ret["body"] = (*this)(_node.body);
|
||||||
return ret;
|
return ret;
|
||||||
@ -139,7 +139,7 @@ Json::Value AsmJsonConverter::operator()(If const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(Switch const& _node) const
|
Json::Value AsmJsonConverter::operator()(Switch const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(_node.expression, "Invalid expression pointer.");
|
yulAssert(_node.expression, "Invalid expression pointer.");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulSwitch");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulSwitch");
|
||||||
ret["expression"] = std::visit(*this, *_node.expression);
|
ret["expression"] = std::visit(*this, *_node.expression);
|
||||||
for (auto const& var: _node.cases)
|
for (auto const& var: _node.cases)
|
||||||
ret["cases"].append((*this)(var));
|
ret["cases"].append((*this)(var));
|
||||||
@ -148,7 +148,7 @@ Json::Value AsmJsonConverter::operator()(Switch const& _node) const
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Case const& _node) const
|
Json::Value AsmJsonConverter::operator()(Case const& _node) const
|
||||||
{
|
{
|
||||||
Json::Value ret = createAstNode(_node.location, "YulCase");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulCase");
|
||||||
ret["value"] = _node.value ? (*this)(*_node.value) : "default";
|
ret["value"] = _node.value ? (*this)(*_node.value) : "default";
|
||||||
ret["body"] = (*this)(_node.body);
|
ret["body"] = (*this)(_node.body);
|
||||||
return ret;
|
return ret;
|
||||||
@ -157,7 +157,7 @@ Json::Value AsmJsonConverter::operator()(Case const& _node) const
|
|||||||
Json::Value AsmJsonConverter::operator()(ForLoop const& _node) const
|
Json::Value AsmJsonConverter::operator()(ForLoop const& _node) const
|
||||||
{
|
{
|
||||||
yulAssert(_node.condition, "Invalid for loop condition.");
|
yulAssert(_node.condition, "Invalid for loop condition.");
|
||||||
Json::Value ret = createAstNode(_node.location, "YulForLoop");
|
Json::Value ret = createAstNode(_node.debugData->location, "YulForLoop");
|
||||||
ret["pre"] = (*this)(_node.pre);
|
ret["pre"] = (*this)(_node.pre);
|
||||||
ret["condition"] = std::visit(*this, *_node.condition);
|
ret["condition"] = std::visit(*this, *_node.condition);
|
||||||
ret["post"] = (*this)(_node.post);
|
ret["post"] = (*this)(_node.post);
|
||||||
@ -167,17 +167,17 @@ Json::Value AsmJsonConverter::operator()(ForLoop const& _node) const
|
|||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Break const& _node) const
|
Json::Value AsmJsonConverter::operator()(Break const& _node) const
|
||||||
{
|
{
|
||||||
return createAstNode(_node.location, "YulBreak");
|
return createAstNode(_node.debugData->location, "YulBreak");
|
||||||
}
|
}
|
||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Continue const& _node) const
|
Json::Value AsmJsonConverter::operator()(Continue const& _node) const
|
||||||
{
|
{
|
||||||
return createAstNode(_node.location, "YulContinue");
|
return createAstNode(_node.debugData->location, "YulContinue");
|
||||||
}
|
}
|
||||||
|
|
||||||
Json::Value AsmJsonConverter::operator()(Leave const& _node) const
|
Json::Value AsmJsonConverter::operator()(Leave const& _node) const
|
||||||
{
|
{
|
||||||
return createAstNode(_node.location, "YulLeave");
|
return createAstNode(_node.debugData->location, "YulLeave");
|
||||||
}
|
}
|
||||||
|
|
||||||
Json::Value AsmJsonConverter::createAstNode(langutil::SourceLocation const& _location, string _nodeType) const
|
Json::Value AsmJsonConverter::createAstNode(langutil::SourceLocation const& _location, string _nodeType) const
|
||||||
|
@ -52,11 +52,12 @@ template <class T>
|
|||||||
T AsmJsonImporter::createAsmNode(Json::Value const& _node)
|
T AsmJsonImporter::createAsmNode(Json::Value const& _node)
|
||||||
{
|
{
|
||||||
T r;
|
T r;
|
||||||
r.location = createSourceLocation(_node);
|
SourceLocation location = createSourceLocation(_node);
|
||||||
yulAssert(
|
yulAssert(
|
||||||
r.location.source && 0 <= r.location.start && r.location.start <= r.location.end,
|
location.source && 0 <= location.start && location.start <= location.end,
|
||||||
"Invalid source location in Asm AST"
|
"Invalid source location in Asm AST"
|
||||||
);
|
);
|
||||||
|
r.debugData = DebugData::create(location);
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,8 +21,8 @@
|
|||||||
* Solidity inline assembly parser.
|
* Solidity inline assembly parser.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AST.h>
|
#include <libyul/AST.h>
|
||||||
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
@ -40,6 +40,21 @@ using namespace solidity::util;
|
|||||||
using namespace solidity::langutil;
|
using namespace solidity::langutil;
|
||||||
using namespace solidity::yul;
|
using namespace solidity::yul;
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
shared_ptr<DebugData const> updateLocationEndFrom(
|
||||||
|
shared_ptr<DebugData const> const& _debugData,
|
||||||
|
langutil::SourceLocation const& _location
|
||||||
|
)
|
||||||
|
{
|
||||||
|
SourceLocation updatedLocation = _debugData->location;
|
||||||
|
updatedLocation.end = _location.end;
|
||||||
|
return make_shared<DebugData const>(updatedLocation);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
unique_ptr<Block> Parser::parse(std::shared_ptr<Scanner> const& _scanner, bool _reuseScanner)
|
unique_ptr<Block> Parser::parse(std::shared_ptr<Scanner> const& _scanner, bool _reuseScanner)
|
||||||
{
|
{
|
||||||
m_recursionDepth = 0;
|
m_recursionDepth = 0;
|
||||||
@ -70,7 +85,7 @@ Block Parser::parseBlock()
|
|||||||
expectToken(Token::LBrace);
|
expectToken(Token::LBrace);
|
||||||
while (currentToken() != Token::RBrace)
|
while (currentToken() != Token::RBrace)
|
||||||
block.statements.emplace_back(parseStatement());
|
block.statements.emplace_back(parseStatement());
|
||||||
block.location.end = currentLocation().end;
|
block.debugData = updateLocationEndFrom(block.debugData, currentLocation());
|
||||||
advance();
|
advance();
|
||||||
return block;
|
return block;
|
||||||
}
|
}
|
||||||
@ -109,7 +124,7 @@ Statement Parser::parseStatement()
|
|||||||
fatalParserError(4904_error, "Case not allowed after default case.");
|
fatalParserError(4904_error, "Case not allowed after default case.");
|
||||||
if (_switch.cases.empty())
|
if (_switch.cases.empty())
|
||||||
fatalParserError(2418_error, "Switch statement without any cases.");
|
fatalParserError(2418_error, "Switch statement without any cases.");
|
||||||
_switch.location.end = _switch.cases.back().body.location.end;
|
_switch.debugData = updateLocationEndFrom(_switch.debugData, _switch.cases.back().body.debugData->location);
|
||||||
return Statement{move(_switch)};
|
return Statement{move(_switch)};
|
||||||
}
|
}
|
||||||
case Token::For:
|
case Token::For:
|
||||||
@ -150,13 +165,13 @@ Statement Parser::parseStatement()
|
|||||||
case Token::LParen:
|
case Token::LParen:
|
||||||
{
|
{
|
||||||
Expression expr = parseCall(std::move(elementary));
|
Expression expr = parseCall(std::move(elementary));
|
||||||
return ExpressionStatement{locationOf(expr), move(expr)};
|
return ExpressionStatement{debugDataOf(expr), move(expr)};
|
||||||
}
|
}
|
||||||
case Token::Comma:
|
case Token::Comma:
|
||||||
case Token::AssemblyAssign:
|
case Token::AssemblyAssign:
|
||||||
{
|
{
|
||||||
Assignment assignment;
|
Assignment assignment;
|
||||||
assignment.location = locationOf(elementary);
|
assignment.debugData = debugDataOf(elementary);
|
||||||
|
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
@ -191,7 +206,7 @@ Statement Parser::parseStatement()
|
|||||||
expectToken(Token::AssemblyAssign);
|
expectToken(Token::AssemblyAssign);
|
||||||
|
|
||||||
assignment.value = make_unique<Expression>(parseExpression());
|
assignment.value = make_unique<Expression>(parseExpression());
|
||||||
assignment.location.end = locationOf(*assignment.value).end;
|
assignment.debugData = updateLocationEndFrom(assignment.debugData, locationOf(*assignment.value));
|
||||||
|
|
||||||
return Statement{move(assignment)};
|
return Statement{move(assignment)};
|
||||||
}
|
}
|
||||||
@ -221,7 +236,7 @@ Case Parser::parseCase()
|
|||||||
else
|
else
|
||||||
yulAssert(false, "Case or default case expected.");
|
yulAssert(false, "Case or default case expected.");
|
||||||
_case.body = parseBlock();
|
_case.body = parseBlock();
|
||||||
_case.location.end = _case.body.location.end;
|
_case.debugData = updateLocationEndFrom(_case.debugData, _case.body.debugData->location);
|
||||||
return _case;
|
return _case;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -241,7 +256,7 @@ ForLoop Parser::parseForLoop()
|
|||||||
forLoop.post = parseBlock();
|
forLoop.post = parseBlock();
|
||||||
m_currentForLoopComponent = ForLoopComponent::ForLoopBody;
|
m_currentForLoopComponent = ForLoopComponent::ForLoopBody;
|
||||||
forLoop.body = parseBlock();
|
forLoop.body = parseBlock();
|
||||||
forLoop.location.end = forLoop.body.location.end;
|
forLoop.debugData = updateLocationEndFrom(forLoop.debugData, forLoop.body.debugData->location);
|
||||||
|
|
||||||
m_currentForLoopComponent = outerForLoopComponent;
|
m_currentForLoopComponent = outerForLoopComponent;
|
||||||
|
|
||||||
@ -261,7 +276,7 @@ Expression Parser::parseExpression()
|
|||||||
if (m_dialect.builtin(_identifier.name))
|
if (m_dialect.builtin(_identifier.name))
|
||||||
fatalParserError(
|
fatalParserError(
|
||||||
7104_error,
|
7104_error,
|
||||||
_identifier.location,
|
_identifier.debugData->location,
|
||||||
"Builtin function \"" + _identifier.name.str() + "\" must be called."
|
"Builtin function \"" + _identifier.name.str() + "\" must be called."
|
||||||
);
|
);
|
||||||
return move(_identifier);
|
return move(_identifier);
|
||||||
@ -280,7 +295,7 @@ variant<Literal, Identifier> Parser::parseLiteralOrIdentifier()
|
|||||||
{
|
{
|
||||||
case Token::Identifier:
|
case Token::Identifier:
|
||||||
{
|
{
|
||||||
Identifier identifier{currentLocation(), YulString{currentLiteral()}};
|
Identifier identifier{DebugData::create(currentLocation()), YulString{currentLiteral()}};
|
||||||
advance();
|
advance();
|
||||||
return identifier;
|
return identifier;
|
||||||
}
|
}
|
||||||
@ -311,7 +326,7 @@ variant<Literal, Identifier> Parser::parseLiteralOrIdentifier()
|
|||||||
}
|
}
|
||||||
|
|
||||||
Literal literal{
|
Literal literal{
|
||||||
currentLocation(),
|
DebugData::create(currentLocation()),
|
||||||
kind,
|
kind,
|
||||||
YulString{currentLiteral()},
|
YulString{currentLiteral()},
|
||||||
kind == LiteralKind::Boolean ? m_dialect.boolType : m_dialect.defaultType
|
kind == LiteralKind::Boolean ? m_dialect.boolType : m_dialect.defaultType
|
||||||
@ -320,7 +335,7 @@ variant<Literal, Identifier> Parser::parseLiteralOrIdentifier()
|
|||||||
if (currentToken() == Token::Colon)
|
if (currentToken() == Token::Colon)
|
||||||
{
|
{
|
||||||
expectToken(Token::Colon);
|
expectToken(Token::Colon);
|
||||||
literal.location.end = currentLocation().end;
|
literal.debugData = updateLocationEndFrom(literal.debugData, currentLocation());
|
||||||
literal.type = expectAsmIdentifier();
|
literal.type = expectAsmIdentifier();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -352,10 +367,11 @@ VariableDeclaration Parser::parseVariableDeclaration()
|
|||||||
{
|
{
|
||||||
expectToken(Token::AssemblyAssign);
|
expectToken(Token::AssemblyAssign);
|
||||||
varDecl.value = make_unique<Expression>(parseExpression());
|
varDecl.value = make_unique<Expression>(parseExpression());
|
||||||
varDecl.location.end = locationOf(*varDecl.value).end;
|
varDecl.debugData = updateLocationEndFrom(varDecl.debugData, locationOf(*varDecl.value));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
varDecl.location.end = varDecl.variables.back().location.end;
|
varDecl.debugData = updateLocationEndFrom(varDecl.debugData, varDecl.variables.back().debugData->location);
|
||||||
|
|
||||||
return varDecl;
|
return varDecl;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -400,7 +416,7 @@ FunctionDefinition Parser::parseFunctionDefinition()
|
|||||||
m_insideFunction = true;
|
m_insideFunction = true;
|
||||||
funDef.body = parseBlock();
|
funDef.body = parseBlock();
|
||||||
m_insideFunction = preInsideFunction;
|
m_insideFunction = preInsideFunction;
|
||||||
funDef.location.end = funDef.body.location.end;
|
funDef.debugData = updateLocationEndFrom(funDef.debugData, funDef.body.debugData->location);
|
||||||
|
|
||||||
m_currentForLoopComponent = outerForLoopComponent;
|
m_currentForLoopComponent = outerForLoopComponent;
|
||||||
return funDef;
|
return funDef;
|
||||||
@ -415,7 +431,7 @@ FunctionCall Parser::parseCall(variant<Literal, Identifier>&& _initialOp)
|
|||||||
|
|
||||||
FunctionCall ret;
|
FunctionCall ret;
|
||||||
ret.functionName = std::move(std::get<Identifier>(_initialOp));
|
ret.functionName = std::move(std::get<Identifier>(_initialOp));
|
||||||
ret.location = ret.functionName.location;
|
ret.debugData = ret.functionName.debugData;
|
||||||
|
|
||||||
expectToken(Token::LParen);
|
expectToken(Token::LParen);
|
||||||
if (currentToken() != Token::RParen)
|
if (currentToken() != Token::RParen)
|
||||||
@ -427,7 +443,7 @@ FunctionCall Parser::parseCall(variant<Literal, Identifier>&& _initialOp)
|
|||||||
ret.arguments.emplace_back(parseExpression());
|
ret.arguments.emplace_back(parseExpression());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ret.location.end = currentLocation().end;
|
ret.debugData = updateLocationEndFrom(ret.debugData, currentLocation());
|
||||||
expectToken(Token::RParen);
|
expectToken(Token::RParen);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -440,7 +456,7 @@ TypedName Parser::parseTypedName()
|
|||||||
if (currentToken() == Token::Colon)
|
if (currentToken() == Token::Colon)
|
||||||
{
|
{
|
||||||
expectToken(Token::Colon);
|
expectToken(Token::Colon);
|
||||||
typedName.location.end = currentLocation().end;
|
typedName.debugData = updateLocationEndFrom(typedName.debugData, currentLocation());
|
||||||
typedName.type = expectAsmIdentifier();
|
typedName.type = expectAsmIdentifier();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -70,7 +70,7 @@ protected:
|
|||||||
template <class T> T createWithLocation() const
|
template <class T> T createWithLocation() const
|
||||||
{
|
{
|
||||||
T r;
|
T r;
|
||||||
r.location = currentLocation();
|
r.debugData = DebugData::create(currentLocation());
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,7 +25,6 @@
|
|||||||
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libyul/backends/evm/EthAssemblyAdapter.h>
|
#include <libyul/backends/evm/EthAssemblyAdapter.h>
|
||||||
#include <libyul/backends/evm/EVMAssembly.h>
|
#include <libyul/backends/evm/EVMAssembly.h>
|
||||||
|
@ -29,17 +29,17 @@ using namespace solidity::langutil;
|
|||||||
Literal Dialect::zeroLiteralForType(solidity::yul::YulString _type) const
|
Literal Dialect::zeroLiteralForType(solidity::yul::YulString _type) const
|
||||||
{
|
{
|
||||||
if (_type == boolType && _type != defaultType)
|
if (_type == boolType && _type != defaultType)
|
||||||
return {SourceLocation{}, LiteralKind::Boolean, "false"_yulstring, _type};
|
return {DebugData::create(), LiteralKind::Boolean, "false"_yulstring, _type};
|
||||||
return {SourceLocation{}, LiteralKind::Number, "0"_yulstring, _type};
|
return {DebugData::create(), LiteralKind::Number, "0"_yulstring, _type};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Literal Dialect::trueLiteral() const
|
Literal Dialect::trueLiteral() const
|
||||||
{
|
{
|
||||||
if (boolType != defaultType)
|
if (boolType != defaultType)
|
||||||
return {SourceLocation{}, LiteralKind::Boolean, "true"_yulstring, boolType};
|
return {DebugData::create(), LiteralKind::Boolean, "true"_yulstring, boolType};
|
||||||
else
|
else
|
||||||
return {SourceLocation{}, LiteralKind::Number, "1"_yulstring, defaultType};
|
return {DebugData::create(), LiteralKind::Number, "1"_yulstring, defaultType};
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Dialect::validTypeForLiteral(LiteralKind _kind, YulString, YulString _type) const
|
bool Dialect::validTypeForLiteral(LiteralKind _kind, YulString, YulString _type) const
|
||||||
|
@ -19,10 +19,10 @@
|
|||||||
* Parser for Yul code and data object container.
|
* Parser for Yul code and data object container.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <libyul/AST.h>
|
||||||
#include <libyul/ObjectParser.h>
|
#include <libyul/ObjectParser.h>
|
||||||
|
|
||||||
#include <libyul/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/AST.h>
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
|
||||||
#include <liblangutil/Token.h>
|
#include <liblangutil/Token.h>
|
||||||
|
@ -53,7 +53,7 @@ bool ScopeFiller::operator()(ExpressionStatement const& _expr)
|
|||||||
bool ScopeFiller::operator()(VariableDeclaration const& _varDecl)
|
bool ScopeFiller::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
for (auto const& variable: _varDecl.variables)
|
for (auto const& variable: _varDecl.variables)
|
||||||
if (!registerVariable(variable, _varDecl.location, *m_currentScope))
|
if (!registerVariable(variable, _varDecl.debugData->location, *m_currentScope))
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -68,7 +68,7 @@ bool ScopeFiller::operator()(FunctionDefinition const& _funDef)
|
|||||||
|
|
||||||
bool success = true;
|
bool success = true;
|
||||||
for (auto const& var: _funDef.parameters + _funDef.returnVariables)
|
for (auto const& var: _funDef.parameters + _funDef.returnVariables)
|
||||||
if (!registerVariable(var, _funDef.location, varScope))
|
if (!registerVariable(var, _funDef.debugData->location, varScope))
|
||||||
success = false;
|
success = false;
|
||||||
|
|
||||||
if (!(*this)(_funDef.body))
|
if (!(*this)(_funDef.body))
|
||||||
@ -162,7 +162,7 @@ bool ScopeFiller::registerFunction(FunctionDefinition const& _funDef)
|
|||||||
//@TODO secondary location
|
//@TODO secondary location
|
||||||
m_errorReporter.declarationError(
|
m_errorReporter.declarationError(
|
||||||
6052_error,
|
6052_error,
|
||||||
_funDef.location,
|
_funDef.debugData->location,
|
||||||
"Function name " + _funDef.name.str() + " already taken in this scope."
|
"Function name " + _funDef.name.str() + " already taken in this scope."
|
||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
|
@ -23,6 +23,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <libyul/ASTForward.h>
|
||||||
|
|
||||||
#include <libsolutil/Common.h>
|
#include <libsolutil/Common.h>
|
||||||
#include <libsolutil/CommonData.h>
|
#include <libsolutil/CommonData.h>
|
||||||
|
|
||||||
|
@ -100,7 +100,7 @@ void ConstantOptimiser::visit(Expression& _e)
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
Expression const* repr =
|
Expression const* repr =
|
||||||
RepresentationFinder(m_dialect, m_meter, locationOf(_e), m_cache)
|
RepresentationFinder(m_dialect, m_meter, debugDataOf(_e), m_cache)
|
||||||
.tryFindRepresentation(valueOfLiteral(literal))
|
.tryFindRepresentation(valueOfLiteral(literal))
|
||||||
)
|
)
|
||||||
_e = ASTCopier{}.translate(*repr);
|
_e = ASTCopier{}.translate(*repr);
|
||||||
@ -180,7 +180,7 @@ Representation const& RepresentationFinder::findRepresentation(u256 const& _valu
|
|||||||
Representation RepresentationFinder::represent(u256 const& _value) const
|
Representation RepresentationFinder::represent(u256 const& _value) const
|
||||||
{
|
{
|
||||||
Representation repr;
|
Representation repr;
|
||||||
repr.expression = make_unique<Expression>(Literal{m_location, LiteralKind::Number, YulString{formatNumber(_value)}, {}});
|
repr.expression = make_unique<Expression>(Literal{m_debugData, LiteralKind::Number, YulString{formatNumber(_value)}, {}});
|
||||||
repr.cost = m_meter.costs(*repr.expression);
|
repr.cost = m_meter.costs(*repr.expression);
|
||||||
return repr;
|
return repr;
|
||||||
}
|
}
|
||||||
@ -192,8 +192,8 @@ Representation RepresentationFinder::represent(
|
|||||||
{
|
{
|
||||||
Representation repr;
|
Representation repr;
|
||||||
repr.expression = make_unique<Expression>(FunctionCall{
|
repr.expression = make_unique<Expression>(FunctionCall{
|
||||||
m_location,
|
m_debugData,
|
||||||
Identifier{m_location, _instruction},
|
Identifier{m_debugData, _instruction},
|
||||||
{ASTCopier{}.translate(*_argument.expression)}
|
{ASTCopier{}.translate(*_argument.expression)}
|
||||||
});
|
});
|
||||||
repr.cost = _argument.cost + m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction);
|
repr.cost = _argument.cost + m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction);
|
||||||
@ -208,8 +208,8 @@ Representation RepresentationFinder::represent(
|
|||||||
{
|
{
|
||||||
Representation repr;
|
Representation repr;
|
||||||
repr.expression = make_unique<Expression>(FunctionCall{
|
repr.expression = make_unique<Expression>(FunctionCall{
|
||||||
m_location,
|
m_debugData,
|
||||||
Identifier{m_location, _instruction},
|
Identifier{m_debugData, _instruction},
|
||||||
{ASTCopier{}.translate(*_arg1.expression), ASTCopier{}.translate(*_arg2.expression)}
|
{ASTCopier{}.translate(*_arg1.expression), ASTCopier{}.translate(*_arg2.expression)}
|
||||||
});
|
});
|
||||||
repr.cost = m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction) + _arg1.cost + _arg2.cost;
|
repr.cost = m_meter.instructionCosts(*m_dialect.builtin(_instruction)->instruction) + _arg1.cost + _arg2.cost;
|
||||||
|
@ -74,12 +74,12 @@ public:
|
|||||||
RepresentationFinder(
|
RepresentationFinder(
|
||||||
EVMDialect const& _dialect,
|
EVMDialect const& _dialect,
|
||||||
GasMeter const& _meter,
|
GasMeter const& _meter,
|
||||||
langutil::SourceLocation _location,
|
std::shared_ptr<DebugData const> _debugData,
|
||||||
std::map<u256, Representation>& _cache
|
std::map<u256, Representation>& _cache
|
||||||
):
|
):
|
||||||
m_dialect(_dialect),
|
m_dialect(_dialect),
|
||||||
m_meter(_meter),
|
m_meter(_meter),
|
||||||
m_location(std::move(_location)),
|
m_debugData(std::move(_debugData)),
|
||||||
m_cache(_cache)
|
m_cache(_cache)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
@ -100,7 +100,7 @@ private:
|
|||||||
|
|
||||||
EVMDialect const& m_dialect;
|
EVMDialect const& m_dialect;
|
||||||
GasMeter const& m_meter;
|
GasMeter const& m_meter;
|
||||||
langutil::SourceLocation m_location;
|
std::shared_ptr<DebugData const> m_debugData;
|
||||||
/// Counter for the complexity of optimization, will stop when it reaches zero.
|
/// Counter for the complexity of optimization, will stop when it reaches zero.
|
||||||
size_t m_maxSteps = 10000;
|
size_t m_maxSteps = 10000;
|
||||||
std::map<u256, Representation>& m_cache;
|
std::map<u256, Representation>& m_cache;
|
||||||
|
@ -39,7 +39,7 @@ size_t constexpr assemblySizeReferenceSize = 4;
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void EVMAssembly::setSourceLocation(SourceLocation const&)
|
void EVMAssembly::setSourceLocation(langutil::SourceLocation const&)
|
||||||
{
|
{
|
||||||
// Ignored for now;
|
// Ignored for now;
|
||||||
}
|
}
|
||||||
|
@ -44,6 +44,16 @@ using namespace solidity;
|
|||||||
using namespace solidity::yul;
|
using namespace solidity::yul;
|
||||||
using namespace solidity::util;
|
using namespace solidity::util;
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
|
||||||
|
langutil::SourceLocation extractSourceLocationFromDebugData(shared_ptr<DebugData const> const& _debugData)
|
||||||
|
{
|
||||||
|
return _debugData ? _debugData->location : langutil::SourceLocation{};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
CodeTransform::CodeTransform(
|
CodeTransform::CodeTransform(
|
||||||
AbstractAssembly& _assembly,
|
AbstractAssembly& _assembly,
|
||||||
AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
@ -145,13 +155,13 @@ void CodeTransform::operator()(VariableDeclaration const& _varDecl)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_varDecl.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_varDecl.debugData));
|
||||||
size_t variablesLeft = numVariables;
|
size_t variablesLeft = numVariables;
|
||||||
while (variablesLeft--)
|
while (variablesLeft--)
|
||||||
m_assembly.appendConstant(u256(0));
|
m_assembly.appendConstant(u256(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_varDecl.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_varDecl.debugData));
|
||||||
bool atTopOfStack = true;
|
bool atTopOfStack = true;
|
||||||
for (size_t varIndex = 0; varIndex < numVariables; ++varIndex)
|
for (size_t varIndex = 0; varIndex < numVariables; ++varIndex)
|
||||||
{
|
{
|
||||||
@ -205,13 +215,13 @@ void CodeTransform::operator()(Assignment const& _assignment)
|
|||||||
std::visit(*this, *_assignment.value);
|
std::visit(*this, *_assignment.value);
|
||||||
expectDeposit(static_cast<int>(_assignment.variableNames.size()), height);
|
expectDeposit(static_cast<int>(_assignment.variableNames.size()), height);
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_assignment.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_assignment.debugData));
|
||||||
generateMultiAssignment(_assignment.variableNames);
|
generateMultiAssignment(_assignment.variableNames);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::operator()(ExpressionStatement const& _statement)
|
void CodeTransform::operator()(ExpressionStatement const& _statement)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_statement.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_statement.debugData));
|
||||||
std::visit(*this, _statement.expression);
|
std::visit(*this, _statement.expression);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -225,7 +235,7 @@ void CodeTransform::operator()(FunctionCall const& _call)
|
|||||||
});
|
});
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_call.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_call.debugData));
|
||||||
EVMAssembly::LabelID returnLabel(numeric_limits<EVMAssembly::LabelID>::max()); // only used for evm 1.0
|
EVMAssembly::LabelID returnLabel(numeric_limits<EVMAssembly::LabelID>::max()); // only used for evm 1.0
|
||||||
|
|
||||||
returnLabel = m_assembly.newLabelId();
|
returnLabel = m_assembly.newLabelId();
|
||||||
@ -240,7 +250,7 @@ void CodeTransform::operator()(FunctionCall const& _call)
|
|||||||
yulAssert(function->arguments.size() == _call.arguments.size(), "");
|
yulAssert(function->arguments.size() == _call.arguments.size(), "");
|
||||||
for (auto const& arg: _call.arguments | ranges::views::reverse)
|
for (auto const& arg: _call.arguments | ranges::views::reverse)
|
||||||
visitExpression(arg);
|
visitExpression(arg);
|
||||||
m_assembly.setSourceLocation(_call.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_call.debugData));
|
||||||
m_assembly.appendJumpTo(
|
m_assembly.appendJumpTo(
|
||||||
functionEntryID(_call.functionName.name, *function),
|
functionEntryID(_call.functionName.name, *function),
|
||||||
static_cast<int>(function->returns.size() - function->arguments.size()) - 1,
|
static_cast<int>(function->returns.size() - function->arguments.size()) - 1,
|
||||||
@ -252,7 +262,7 @@ void CodeTransform::operator()(FunctionCall const& _call)
|
|||||||
|
|
||||||
void CodeTransform::operator()(Identifier const& _identifier)
|
void CodeTransform::operator()(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_identifier.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_identifier.debugData));
|
||||||
// First search internals, then externals.
|
// First search internals, then externals.
|
||||||
yulAssert(m_scope, "");
|
yulAssert(m_scope, "");
|
||||||
if (m_scope->lookup(_identifier.name, GenericVisitor{
|
if (m_scope->lookup(_identifier.name, GenericVisitor{
|
||||||
@ -284,19 +294,19 @@ void CodeTransform::operator()(Identifier const& _identifier)
|
|||||||
|
|
||||||
void CodeTransform::operator()(Literal const& _literal)
|
void CodeTransform::operator()(Literal const& _literal)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_literal.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_literal.debugData));
|
||||||
m_assembly.appendConstant(valueOfLiteral(_literal));
|
m_assembly.appendConstant(valueOfLiteral(_literal));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::operator()(If const& _if)
|
void CodeTransform::operator()(If const& _if)
|
||||||
{
|
{
|
||||||
visitExpression(*_if.condition);
|
visitExpression(*_if.condition);
|
||||||
m_assembly.setSourceLocation(_if.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_if.debugData));
|
||||||
m_assembly.appendInstruction(evmasm::Instruction::ISZERO);
|
m_assembly.appendInstruction(evmasm::Instruction::ISZERO);
|
||||||
AbstractAssembly::LabelID end = m_assembly.newLabelId();
|
AbstractAssembly::LabelID end = m_assembly.newLabelId();
|
||||||
m_assembly.appendJumpToIf(end);
|
m_assembly.appendJumpToIf(end);
|
||||||
(*this)(_if.body);
|
(*this)(_if.body);
|
||||||
m_assembly.setSourceLocation(_if.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_if.debugData));
|
||||||
m_assembly.appendLabel(end);
|
m_assembly.appendLabel(end);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,7 +323,7 @@ void CodeTransform::operator()(Switch const& _switch)
|
|||||||
if (c.value)
|
if (c.value)
|
||||||
{
|
{
|
||||||
(*this)(*c.value);
|
(*this)(*c.value);
|
||||||
m_assembly.setSourceLocation(c.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(c.debugData));
|
||||||
AbstractAssembly::LabelID bodyLabel = m_assembly.newLabelId();
|
AbstractAssembly::LabelID bodyLabel = m_assembly.newLabelId();
|
||||||
caseBodies[&c] = bodyLabel;
|
caseBodies[&c] = bodyLabel;
|
||||||
yulAssert(m_assembly.stackHeight() == expressionHeight + 1, "");
|
yulAssert(m_assembly.stackHeight() == expressionHeight + 1, "");
|
||||||
@ -325,24 +335,24 @@ void CodeTransform::operator()(Switch const& _switch)
|
|||||||
// default case
|
// default case
|
||||||
(*this)(c.body);
|
(*this)(c.body);
|
||||||
}
|
}
|
||||||
m_assembly.setSourceLocation(_switch.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_switch.debugData));
|
||||||
m_assembly.appendJumpTo(end);
|
m_assembly.appendJumpTo(end);
|
||||||
|
|
||||||
size_t numCases = caseBodies.size();
|
size_t numCases = caseBodies.size();
|
||||||
for (auto const& c: caseBodies)
|
for (auto const& c: caseBodies)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(c.first->location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(c.first->debugData));
|
||||||
m_assembly.appendLabel(c.second);
|
m_assembly.appendLabel(c.second);
|
||||||
(*this)(c.first->body);
|
(*this)(c.first->body);
|
||||||
// Avoid useless "jump to next" for the last case.
|
// Avoid useless "jump to next" for the last case.
|
||||||
if (--numCases > 0)
|
if (--numCases > 0)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(c.first->location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(c.first->debugData));
|
||||||
m_assembly.appendJumpTo(end);
|
m_assembly.appendJumpTo(end);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_switch.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_switch.debugData));
|
||||||
m_assembly.appendLabel(end);
|
m_assembly.appendLabel(end);
|
||||||
m_assembly.appendInstruction(evmasm::Instruction::POP);
|
m_assembly.appendInstruction(evmasm::Instruction::POP);
|
||||||
}
|
}
|
||||||
@ -363,7 +373,7 @@ void CodeTransform::operator()(FunctionDefinition const& _function)
|
|||||||
m_context->variableStackHeights[&var] = height++;
|
m_context->variableStackHeights[&var] = height++;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_function.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_function.debugData));
|
||||||
int const stackHeightBefore = m_assembly.stackHeight();
|
int const stackHeightBefore = m_assembly.stackHeight();
|
||||||
|
|
||||||
m_assembly.appendLabel(functionEntryID(_function.name, function));
|
m_assembly.appendLabel(functionEntryID(_function.name, function));
|
||||||
@ -488,11 +498,11 @@ void CodeTransform::operator()(ForLoop const& _forLoop)
|
|||||||
AbstractAssembly::LabelID postPart = m_assembly.newLabelId();
|
AbstractAssembly::LabelID postPart = m_assembly.newLabelId();
|
||||||
AbstractAssembly::LabelID loopEnd = m_assembly.newLabelId();
|
AbstractAssembly::LabelID loopEnd = m_assembly.newLabelId();
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_forLoop.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_forLoop.debugData));
|
||||||
m_assembly.appendLabel(loopStart);
|
m_assembly.appendLabel(loopStart);
|
||||||
|
|
||||||
visitExpression(*_forLoop.condition);
|
visitExpression(*_forLoop.condition);
|
||||||
m_assembly.setSourceLocation(_forLoop.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_forLoop.debugData));
|
||||||
m_assembly.appendInstruction(evmasm::Instruction::ISZERO);
|
m_assembly.appendInstruction(evmasm::Instruction::ISZERO);
|
||||||
m_assembly.appendJumpToIf(loopEnd);
|
m_assembly.appendJumpToIf(loopEnd);
|
||||||
|
|
||||||
@ -500,12 +510,12 @@ void CodeTransform::operator()(ForLoop const& _forLoop)
|
|||||||
m_context->forLoopStack.emplace(Context::ForLoopLabels{ {postPart, stackHeightBody}, {loopEnd, stackHeightBody} });
|
m_context->forLoopStack.emplace(Context::ForLoopLabels{ {postPart, stackHeightBody}, {loopEnd, stackHeightBody} });
|
||||||
(*this)(_forLoop.body);
|
(*this)(_forLoop.body);
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_forLoop.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_forLoop.debugData));
|
||||||
m_assembly.appendLabel(postPart);
|
m_assembly.appendLabel(postPart);
|
||||||
|
|
||||||
(*this)(_forLoop.post);
|
(*this)(_forLoop.post);
|
||||||
|
|
||||||
m_assembly.setSourceLocation(_forLoop.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_forLoop.debugData));
|
||||||
m_assembly.appendJumpTo(loopStart);
|
m_assembly.appendJumpTo(loopStart);
|
||||||
m_assembly.appendLabel(loopEnd);
|
m_assembly.appendLabel(loopEnd);
|
||||||
|
|
||||||
@ -525,7 +535,7 @@ int CodeTransform::appendPopUntil(int _targetDepth)
|
|||||||
void CodeTransform::operator()(Break const& _break)
|
void CodeTransform::operator()(Break const& _break)
|
||||||
{
|
{
|
||||||
yulAssert(!m_context->forLoopStack.empty(), "Invalid break-statement. Requires surrounding for-loop in code generation.");
|
yulAssert(!m_context->forLoopStack.empty(), "Invalid break-statement. Requires surrounding for-loop in code generation.");
|
||||||
m_assembly.setSourceLocation(_break.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_break.debugData));
|
||||||
|
|
||||||
Context::JumpInfo const& jump = m_context->forLoopStack.top().done;
|
Context::JumpInfo const& jump = m_context->forLoopStack.top().done;
|
||||||
m_assembly.appendJumpTo(jump.label, appendPopUntil(jump.targetStackHeight));
|
m_assembly.appendJumpTo(jump.label, appendPopUntil(jump.targetStackHeight));
|
||||||
@ -534,7 +544,7 @@ void CodeTransform::operator()(Break const& _break)
|
|||||||
void CodeTransform::operator()(Continue const& _continue)
|
void CodeTransform::operator()(Continue const& _continue)
|
||||||
{
|
{
|
||||||
yulAssert(!m_context->forLoopStack.empty(), "Invalid continue-statement. Requires surrounding for-loop in code generation.");
|
yulAssert(!m_context->forLoopStack.empty(), "Invalid continue-statement. Requires surrounding for-loop in code generation.");
|
||||||
m_assembly.setSourceLocation(_continue.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_continue.debugData));
|
||||||
|
|
||||||
Context::JumpInfo const& jump = m_context->forLoopStack.top().post;
|
Context::JumpInfo const& jump = m_context->forLoopStack.top().post;
|
||||||
m_assembly.appendJumpTo(jump.label, appendPopUntil(jump.targetStackHeight));
|
m_assembly.appendJumpTo(jump.label, appendPopUntil(jump.targetStackHeight));
|
||||||
@ -544,7 +554,7 @@ void CodeTransform::operator()(Leave const& _leaveStatement)
|
|||||||
{
|
{
|
||||||
yulAssert(m_functionExitLabel, "Invalid leave-statement. Requires surrounding function in code generation.");
|
yulAssert(m_functionExitLabel, "Invalid leave-statement. Requires surrounding function in code generation.");
|
||||||
yulAssert(m_functionExitStackHeight, "");
|
yulAssert(m_functionExitStackHeight, "");
|
||||||
m_assembly.setSourceLocation(_leaveStatement.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_leaveStatement.debugData));
|
||||||
m_assembly.appendJumpTo(*m_functionExitLabel, appendPopUntil(*m_functionExitStackHeight));
|
m_assembly.appendJumpTo(*m_functionExitLabel, appendPopUntil(*m_functionExitStackHeight));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -605,7 +615,7 @@ void CodeTransform::setupReturnVariablesAndFunctionExit()
|
|||||||
|
|
||||||
// Allocate slots for return variables as if they were declared as variables in the virtual function scope.
|
// Allocate slots for return variables as if they were declared as variables in the virtual function scope.
|
||||||
for (TypedName const& var: m_delayedReturnVariables)
|
for (TypedName const& var: m_delayedReturnVariables)
|
||||||
(*this)(VariableDeclaration{var.location, {var}, {}});
|
(*this)(VariableDeclaration{var.debugData, {var}, {}});
|
||||||
|
|
||||||
m_functionExitStackHeight = ranges::max(m_delayedReturnVariables | ranges::views::transform([&](TypedName const& _name) {
|
m_functionExitStackHeight = ranges::max(m_delayedReturnVariables | ranges::views::transform([&](TypedName const& _name) {
|
||||||
return variableStackHeight(_name.name);
|
return variableStackHeight(_name.name);
|
||||||
@ -654,7 +664,7 @@ void CodeTransform::visitStatements(vector<Statement> const& _statements)
|
|||||||
auto const* functionDefinition = std::get_if<FunctionDefinition>(&statement);
|
auto const* functionDefinition = std::get_if<FunctionDefinition>(&statement);
|
||||||
if (functionDefinition && !jumpTarget)
|
if (functionDefinition && !jumpTarget)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(locationOf(statement));
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(functionDefinition->debugData));
|
||||||
jumpTarget = m_assembly.newLabelId();
|
jumpTarget = m_assembly.newLabelId();
|
||||||
m_assembly.appendJumpTo(*jumpTarget, 0);
|
m_assembly.appendJumpTo(*jumpTarget, 0);
|
||||||
}
|
}
|
||||||
@ -675,7 +685,7 @@ void CodeTransform::visitStatements(vector<Statement> const& _statements)
|
|||||||
|
|
||||||
void CodeTransform::finalizeBlock(Block const& _block, optional<int> blockStartStackHeight)
|
void CodeTransform::finalizeBlock(Block const& _block, optional<int> blockStartStackHeight)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_block.location);
|
m_assembly.setSourceLocation(extractSourceLocationFromDebugData(_block.debugData));
|
||||||
|
|
||||||
freeUnusedVariables();
|
freeUnusedVariables();
|
||||||
|
|
||||||
|
@ -39,6 +39,7 @@ class ErrorReporter;
|
|||||||
|
|
||||||
namespace solidity::yul
|
namespace solidity::yul
|
||||||
{
|
{
|
||||||
|
|
||||||
struct AsmAnalysisInfo;
|
struct AsmAnalysisInfo;
|
||||||
class EVMAssembly;
|
class EVMAssembly;
|
||||||
|
|
||||||
|
@ -55,7 +55,7 @@ void visitArguments(
|
|||||||
for (auto const& arg: _call.arguments | ranges::views::reverse)
|
for (auto const& arg: _call.arguments | ranges::views::reverse)
|
||||||
_visitExpression(arg);
|
_visitExpression(arg);
|
||||||
|
|
||||||
_assembly.setSourceLocation(_call.location);
|
_assembly.setSourceLocation(_call.debugData->location);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -259,7 +259,7 @@ map<YulString, BuiltinFunctionForEVM> createBuiltins(langutil::EVMVersion _evmVe
|
|||||||
_visitExpression(_call.arguments[2]);
|
_visitExpression(_call.arguments[2]);
|
||||||
YulString identifier = std::get<Literal>(_call.arguments[1]).value;
|
YulString identifier = std::get<Literal>(_call.arguments[1]).value;
|
||||||
_visitExpression(_call.arguments[0]);
|
_visitExpression(_call.arguments[0]);
|
||||||
_assembly.setSourceLocation(_call.location);
|
_assembly.setSourceLocation(_call.debugData->location);
|
||||||
_assembly.appendImmutableAssignment(identifier.str());
|
_assembly.appendImmutableAssignment(identifier.str());
|
||||||
}
|
}
|
||||||
));
|
));
|
||||||
|
@ -32,10 +32,10 @@
|
|||||||
#include <libyul/optimiser/OptimiserStep.h>
|
#include <libyul/optimiser/OptimiserStep.h>
|
||||||
#include <libyul/optimiser/ForLoopConditionIntoBody.h>
|
#include <libyul/optimiser/ForLoopConditionIntoBody.h>
|
||||||
|
|
||||||
|
#include <libyul/AST.h>
|
||||||
#include <libyul/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AST.h>
|
|
||||||
#include <libyul/Object.h>
|
#include <libyul/Object.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
@ -65,8 +65,8 @@ void WordSizeTransform::operator()(FunctionCall& _fc)
|
|||||||
void WordSizeTransform::operator()(If& _if)
|
void WordSizeTransform::operator()(If& _if)
|
||||||
{
|
{
|
||||||
_if.condition = make_unique<Expression>(FunctionCall{
|
_if.condition = make_unique<Expression>(FunctionCall{
|
||||||
locationOf(*_if.condition),
|
debugDataOf(*_if.condition),
|
||||||
Identifier{locationOf(*_if.condition), "or_bool"_yulstring},
|
Identifier{debugDataOf(*_if.condition), "or_bool"_yulstring},
|
||||||
expandValueToVector(*_if.condition)
|
expandValueToVector(*_if.condition)
|
||||||
});
|
});
|
||||||
(*this)(_if.body);
|
(*this)(_if.body);
|
||||||
@ -81,8 +81,8 @@ void WordSizeTransform::operator()(ForLoop& _for)
|
|||||||
{
|
{
|
||||||
(*this)(_for.pre);
|
(*this)(_for.pre);
|
||||||
_for.condition = make_unique<Expression>(FunctionCall{
|
_for.condition = make_unique<Expression>(FunctionCall{
|
||||||
locationOf(*_for.condition),
|
debugDataOf(*_for.condition),
|
||||||
Identifier{locationOf(*_for.condition), "or_bool"_yulstring},
|
Identifier{debugDataOf(*_for.condition), "or_bool"_yulstring},
|
||||||
expandValueToVector(*_for.condition)
|
expandValueToVector(*_for.condition)
|
||||||
});
|
});
|
||||||
(*this)(_for.post);
|
(*this)(_for.post);
|
||||||
@ -116,18 +116,18 @@ void WordSizeTransform::operator()(Block& _block)
|
|||||||
vector<Statement> ret;
|
vector<Statement> ret;
|
||||||
for (size_t i = 0; i < 3; i++)
|
for (size_t i = 0; i < 3; i++)
|
||||||
ret.emplace_back(VariableDeclaration{
|
ret.emplace_back(VariableDeclaration{
|
||||||
varDecl.location,
|
varDecl.debugData,
|
||||||
{TypedName{varDecl.location, newLhs[i], m_targetDialect.defaultType}},
|
{TypedName{varDecl.debugData, newLhs[i], m_targetDialect.defaultType}},
|
||||||
make_unique<Expression>(Literal{
|
make_unique<Expression>(Literal{
|
||||||
locationOf(*varDecl.value),
|
debugDataOf(*varDecl.value),
|
||||||
LiteralKind::Number,
|
LiteralKind::Number,
|
||||||
"0"_yulstring,
|
"0"_yulstring,
|
||||||
m_targetDialect.defaultType
|
m_targetDialect.defaultType
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
ret.emplace_back(VariableDeclaration{
|
ret.emplace_back(VariableDeclaration{
|
||||||
varDecl.location,
|
varDecl.debugData,
|
||||||
{TypedName{varDecl.location, newLhs[3], m_targetDialect.defaultType}},
|
{TypedName{varDecl.debugData, newLhs[3], m_targetDialect.defaultType}},
|
||||||
std::move(varDecl.value)
|
std::move(varDecl.value)
|
||||||
});
|
});
|
||||||
return {std::move(ret)};
|
return {std::move(ret)};
|
||||||
@ -147,8 +147,8 @@ void WordSizeTransform::operator()(Block& _block)
|
|||||||
vector<Statement> ret;
|
vector<Statement> ret;
|
||||||
for (size_t i = 0; i < 4; i++)
|
for (size_t i = 0; i < 4; i++)
|
||||||
ret.emplace_back(VariableDeclaration{
|
ret.emplace_back(VariableDeclaration{
|
||||||
varDecl.location,
|
varDecl.debugData,
|
||||||
{TypedName{varDecl.location, newLhs[i], m_targetDialect.defaultType}},
|
{TypedName{varDecl.debugData, newLhs[i], m_targetDialect.defaultType}},
|
||||||
std::move(newRhs[i])
|
std::move(newRhs[i])
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -177,18 +177,18 @@ void WordSizeTransform::operator()(Block& _block)
|
|||||||
vector<Statement> ret;
|
vector<Statement> ret;
|
||||||
for (size_t i = 0; i < 3; i++)
|
for (size_t i = 0; i < 3; i++)
|
||||||
ret.emplace_back(Assignment{
|
ret.emplace_back(Assignment{
|
||||||
assignment.location,
|
assignment.debugData,
|
||||||
{Identifier{assignment.location, newLhs[i]}},
|
{Identifier{assignment.debugData, newLhs[i]}},
|
||||||
make_unique<Expression>(Literal{
|
make_unique<Expression>(Literal{
|
||||||
locationOf(*assignment.value),
|
debugDataOf(*assignment.value),
|
||||||
LiteralKind::Number,
|
LiteralKind::Number,
|
||||||
"0"_yulstring,
|
"0"_yulstring,
|
||||||
m_targetDialect.defaultType
|
m_targetDialect.defaultType
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
ret.emplace_back(Assignment{
|
ret.emplace_back(Assignment{
|
||||||
assignment.location,
|
assignment.debugData,
|
||||||
{Identifier{assignment.location, newLhs[3]}},
|
{Identifier{assignment.debugData, newLhs[3]}},
|
||||||
std::move(assignment.value)
|
std::move(assignment.value)
|
||||||
});
|
});
|
||||||
return {std::move(ret)};
|
return {std::move(ret)};
|
||||||
@ -208,8 +208,8 @@ void WordSizeTransform::operator()(Block& _block)
|
|||||||
vector<Statement> ret;
|
vector<Statement> ret;
|
||||||
for (size_t i = 0; i < 4; i++)
|
for (size_t i = 0; i < 4; i++)
|
||||||
ret.emplace_back(Assignment{
|
ret.emplace_back(Assignment{
|
||||||
assignment.location,
|
assignment.debugData,
|
||||||
{Identifier{assignment.location, m_variableMapping.at(lhsName)[i]}},
|
{Identifier{assignment.debugData, m_variableMapping.at(lhsName)[i]}},
|
||||||
std::move(newRhs[i])
|
std::move(newRhs[i])
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -258,7 +258,7 @@ void WordSizeTransform::rewriteVarDeclList(TypedNameList& _nameList)
|
|||||||
{
|
{
|
||||||
TypedNameList ret;
|
TypedNameList ret;
|
||||||
for (auto newName: generateU64IdentifierNames(_n.name))
|
for (auto newName: generateU64IdentifierNames(_n.name))
|
||||||
ret.emplace_back(TypedName{_n.location, newName, m_targetDialect.defaultType});
|
ret.emplace_back(TypedName{_n.debugData, newName, m_targetDialect.defaultType});
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -272,14 +272,14 @@ void WordSizeTransform::rewriteIdentifierList(vector<Identifier>& _ids)
|
|||||||
{
|
{
|
||||||
vector<Identifier> ret;
|
vector<Identifier> ret;
|
||||||
for (auto newId: m_variableMapping.at(_id.name))
|
for (auto newId: m_variableMapping.at(_id.name))
|
||||||
ret.push_back(Identifier{_id.location, newId});
|
ret.push_back(Identifier{_id.debugData, newId});
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<Statement> WordSizeTransform::handleSwitchInternal(
|
vector<Statement> WordSizeTransform::handleSwitchInternal(
|
||||||
langutil::SourceLocation const& _location,
|
shared_ptr<DebugData const> const& _debugData,
|
||||||
vector<YulString> const& _splitExpressions,
|
vector<YulString> const& _splitExpressions,
|
||||||
vector<Case> _cases,
|
vector<Case> _cases,
|
||||||
YulString _runDefaultFlag,
|
YulString _runDefaultFlag,
|
||||||
@ -304,19 +304,19 @@ vector<Statement> WordSizeTransform::handleSwitchInternal(
|
|||||||
}
|
}
|
||||||
|
|
||||||
Switch ret{
|
Switch ret{
|
||||||
_location,
|
_debugData,
|
||||||
make_unique<Expression>(Identifier{_location, _splitExpressions.at(_depth)}),
|
make_unique<Expression>(Identifier{_debugData, _splitExpressions.at(_depth)}),
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
||||||
for (auto& c: cases)
|
for (auto& c: cases)
|
||||||
{
|
{
|
||||||
Literal label{_location, LiteralKind::Number, YulString(c.first.str()), m_targetDialect.defaultType};
|
Literal label{_debugData, LiteralKind::Number, YulString(c.first.str()), m_targetDialect.defaultType};
|
||||||
ret.cases.emplace_back(Case{
|
ret.cases.emplace_back(Case{
|
||||||
c.second.front().location,
|
c.second.front().debugData,
|
||||||
make_unique<Literal>(std::move(label)),
|
make_unique<Literal>(std::move(label)),
|
||||||
Block{_location, handleSwitchInternal(
|
Block{_debugData, handleSwitchInternal(
|
||||||
_location,
|
_debugData,
|
||||||
_splitExpressions,
|
_splitExpressions,
|
||||||
std::move(c.second),
|
std::move(c.second),
|
||||||
_runDefaultFlag,
|
_runDefaultFlag,
|
||||||
@ -326,13 +326,13 @@ vector<Statement> WordSizeTransform::handleSwitchInternal(
|
|||||||
}
|
}
|
||||||
if (!_runDefaultFlag.empty())
|
if (!_runDefaultFlag.empty())
|
||||||
ret.cases.emplace_back(Case{
|
ret.cases.emplace_back(Case{
|
||||||
_location,
|
_debugData,
|
||||||
nullptr,
|
nullptr,
|
||||||
Block{_location, make_vector<Statement>(
|
Block{_debugData, make_vector<Statement>(
|
||||||
Assignment{
|
Assignment{
|
||||||
_location,
|
_debugData,
|
||||||
{{_location, _runDefaultFlag}},
|
{{_debugData, _runDefaultFlag}},
|
||||||
make_unique<Expression>(Literal{_location, LiteralKind::Boolean, "true"_yulstring, m_targetDialect.boolType})
|
make_unique<Expression>(Literal{_debugData, LiteralKind::Boolean, "true"_yulstring, m_targetDialect.boolType})
|
||||||
}
|
}
|
||||||
)}
|
)}
|
||||||
});
|
});
|
||||||
@ -356,8 +356,8 @@ std::vector<Statement> WordSizeTransform::handleSwitch(Switch& _switch)
|
|||||||
defaultCase = std::move(_switch.cases.back());
|
defaultCase = std::move(_switch.cases.back());
|
||||||
_switch.cases.pop_back();
|
_switch.cases.pop_back();
|
||||||
ret.emplace_back(VariableDeclaration{
|
ret.emplace_back(VariableDeclaration{
|
||||||
_switch.location,
|
_switch.debugData,
|
||||||
{TypedName{_switch.location, runDefaultFlag, m_targetDialect.boolType}},
|
{TypedName{_switch.debugData, runDefaultFlag, m_targetDialect.boolType}},
|
||||||
{}
|
{}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -366,7 +366,7 @@ std::vector<Statement> WordSizeTransform::handleSwitch(Switch& _switch)
|
|||||||
splitExpressions.emplace_back(std::get<Identifier>(*expr).name);
|
splitExpressions.emplace_back(std::get<Identifier>(*expr).name);
|
||||||
|
|
||||||
ret += handleSwitchInternal(
|
ret += handleSwitchInternal(
|
||||||
_switch.location,
|
_switch.debugData,
|
||||||
splitExpressions,
|
splitExpressions,
|
||||||
std::move(_switch.cases),
|
std::move(_switch.cases),
|
||||||
runDefaultFlag,
|
runDefaultFlag,
|
||||||
@ -374,8 +374,8 @@ std::vector<Statement> WordSizeTransform::handleSwitch(Switch& _switch)
|
|||||||
);
|
);
|
||||||
if (!runDefaultFlag.empty())
|
if (!runDefaultFlag.empty())
|
||||||
ret.emplace_back(If{
|
ret.emplace_back(If{
|
||||||
_switch.location,
|
_switch.debugData,
|
||||||
make_unique<Expression>(Identifier{_switch.location, runDefaultFlag}),
|
make_unique<Expression>(Identifier{_switch.debugData, runDefaultFlag}),
|
||||||
std::move(defaultCase.body)
|
std::move(defaultCase.body)
|
||||||
});
|
});
|
||||||
return ret;
|
return ret;
|
||||||
@ -397,7 +397,7 @@ array<unique_ptr<Expression>, 4> WordSizeTransform::expandValue(Expression const
|
|||||||
{
|
{
|
||||||
auto const& id = std::get<Identifier>(_e);
|
auto const& id = std::get<Identifier>(_e);
|
||||||
for (size_t i = 0; i < 4; i++)
|
for (size_t i = 0; i < 4; i++)
|
||||||
ret[i] = make_unique<Expression>(Identifier{id.location, m_variableMapping.at(id.name)[i]});
|
ret[i] = make_unique<Expression>(Identifier{id.debugData, m_variableMapping.at(id.name)[i]});
|
||||||
}
|
}
|
||||||
else if (holds_alternative<Literal>(_e))
|
else if (holds_alternative<Literal>(_e))
|
||||||
{
|
{
|
||||||
@ -410,7 +410,7 @@ array<unique_ptr<Expression>, 4> WordSizeTransform::expandValue(Expression const
|
|||||||
val >>= 64;
|
val >>= 64;
|
||||||
ret[exprIndexReverse] = make_unique<Expression>(
|
ret[exprIndexReverse] = make_unique<Expression>(
|
||||||
Literal{
|
Literal{
|
||||||
lit.location,
|
lit.debugData,
|
||||||
LiteralKind::Number,
|
LiteralKind::Number,
|
||||||
YulString(currentVal.str()),
|
YulString(currentVal.str()),
|
||||||
m_targetDialect.defaultType
|
m_targetDialect.defaultType
|
||||||
|
@ -87,7 +87,7 @@ private:
|
|||||||
|
|
||||||
std::vector<Statement> handleSwitch(Switch& _switch);
|
std::vector<Statement> handleSwitch(Switch& _switch);
|
||||||
std::vector<Statement> handleSwitchInternal(
|
std::vector<Statement> handleSwitchInternal(
|
||||||
langutil::SourceLocation const& _location,
|
std::shared_ptr<DebugData const> const& _debugData,
|
||||||
std::vector<YulString> const& _splitExpressions,
|
std::vector<YulString> const& _splitExpressions,
|
||||||
std::vector<Case> _cases,
|
std::vector<Case> _cases,
|
||||||
YulString _runDefaultFlag,
|
YulString _runDefaultFlag,
|
||||||
|
@ -34,13 +34,13 @@ using namespace solidity::util;
|
|||||||
|
|
||||||
Statement ASTCopier::operator()(ExpressionStatement const& _statement)
|
Statement ASTCopier::operator()(ExpressionStatement const& _statement)
|
||||||
{
|
{
|
||||||
return ExpressionStatement{ _statement.location, translate(_statement.expression) };
|
return ExpressionStatement{ _statement.debugData, translate(_statement.expression) };
|
||||||
}
|
}
|
||||||
|
|
||||||
Statement ASTCopier::operator()(VariableDeclaration const& _varDecl)
|
Statement ASTCopier::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
return VariableDeclaration{
|
return VariableDeclaration{
|
||||||
_varDecl.location,
|
_varDecl.debugData,
|
||||||
translateVector(_varDecl.variables),
|
translateVector(_varDecl.variables),
|
||||||
translate(_varDecl.value)
|
translate(_varDecl.value)
|
||||||
};
|
};
|
||||||
@ -49,7 +49,7 @@ Statement ASTCopier::operator()(VariableDeclaration const& _varDecl)
|
|||||||
Statement ASTCopier::operator()(Assignment const& _assignment)
|
Statement ASTCopier::operator()(Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
return Assignment{
|
return Assignment{
|
||||||
_assignment.location,
|
_assignment.debugData,
|
||||||
translateVector(_assignment.variableNames),
|
translateVector(_assignment.variableNames),
|
||||||
translate(_assignment.value)
|
translate(_assignment.value)
|
||||||
};
|
};
|
||||||
@ -58,7 +58,7 @@ Statement ASTCopier::operator()(Assignment const& _assignment)
|
|||||||
Expression ASTCopier::operator()(FunctionCall const& _call)
|
Expression ASTCopier::operator()(FunctionCall const& _call)
|
||||||
{
|
{
|
||||||
return FunctionCall{
|
return FunctionCall{
|
||||||
_call.location,
|
_call.debugData,
|
||||||
translate(_call.functionName),
|
translate(_call.functionName),
|
||||||
translateVector(_call.arguments)
|
translateVector(_call.arguments)
|
||||||
};
|
};
|
||||||
@ -76,12 +76,12 @@ Expression ASTCopier::operator()(Literal const& _literal)
|
|||||||
|
|
||||||
Statement ASTCopier::operator()(If const& _if)
|
Statement ASTCopier::operator()(If const& _if)
|
||||||
{
|
{
|
||||||
return If{_if.location, translate(_if.condition), translate(_if.body)};
|
return If{_if.debugData, translate(_if.condition), translate(_if.body)};
|
||||||
}
|
}
|
||||||
|
|
||||||
Statement ASTCopier::operator()(Switch const& _switch)
|
Statement ASTCopier::operator()(Switch const& _switch)
|
||||||
{
|
{
|
||||||
return Switch{_switch.location, translate(_switch.expression), translateVector(_switch.cases)};
|
return Switch{_switch.debugData, translate(_switch.expression), translateVector(_switch.cases)};
|
||||||
}
|
}
|
||||||
|
|
||||||
Statement ASTCopier::operator()(FunctionDefinition const& _function)
|
Statement ASTCopier::operator()(FunctionDefinition const& _function)
|
||||||
@ -92,7 +92,7 @@ Statement ASTCopier::operator()(FunctionDefinition const& _function)
|
|||||||
ScopeGuard g([&]() { this->leaveFunction(_function); });
|
ScopeGuard g([&]() { this->leaveFunction(_function); });
|
||||||
|
|
||||||
return FunctionDefinition{
|
return FunctionDefinition{
|
||||||
_function.location,
|
_function.debugData,
|
||||||
translatedName,
|
translatedName,
|
||||||
translateVector(_function.parameters),
|
translateVector(_function.parameters),
|
||||||
translateVector(_function.returnVariables),
|
translateVector(_function.returnVariables),
|
||||||
@ -106,7 +106,7 @@ Statement ASTCopier::operator()(ForLoop const& _forLoop)
|
|||||||
ScopeGuard g([&]() { this->leaveScope(_forLoop.pre); });
|
ScopeGuard g([&]() { this->leaveScope(_forLoop.pre); });
|
||||||
|
|
||||||
return ForLoop{
|
return ForLoop{
|
||||||
_forLoop.location,
|
_forLoop.debugData,
|
||||||
translate(_forLoop.pre),
|
translate(_forLoop.pre),
|
||||||
translate(_forLoop.condition),
|
translate(_forLoop.condition),
|
||||||
translate(_forLoop.post),
|
translate(_forLoop.post),
|
||||||
@ -148,17 +148,17 @@ Block ASTCopier::translate(Block const& _block)
|
|||||||
enterScope(_block);
|
enterScope(_block);
|
||||||
ScopeGuard g([&]() { this->leaveScope(_block); });
|
ScopeGuard g([&]() { this->leaveScope(_block); });
|
||||||
|
|
||||||
return Block{_block.location, translateVector(_block.statements)};
|
return Block{_block.debugData, translateVector(_block.statements)};
|
||||||
}
|
}
|
||||||
|
|
||||||
Case ASTCopier::translate(Case const& _case)
|
Case ASTCopier::translate(Case const& _case)
|
||||||
{
|
{
|
||||||
return Case{_case.location, translate(_case.value), translate(_case.body)};
|
return Case{_case.debugData, translate(_case.value), translate(_case.body)};
|
||||||
}
|
}
|
||||||
|
|
||||||
Identifier ASTCopier::translate(Identifier const& _identifier)
|
Identifier ASTCopier::translate(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
return Identifier{_identifier.location, translateIdentifier(_identifier.name)};
|
return Identifier{_identifier.debugData, translateIdentifier(_identifier.name)};
|
||||||
}
|
}
|
||||||
|
|
||||||
Literal ASTCopier::translate(Literal const& _literal)
|
Literal ASTCopier::translate(Literal const& _literal)
|
||||||
@ -168,7 +168,7 @@ Literal ASTCopier::translate(Literal const& _literal)
|
|||||||
|
|
||||||
TypedName ASTCopier::translate(TypedName const& _typedName)
|
TypedName ASTCopier::translate(TypedName const& _typedName)
|
||||||
{
|
{
|
||||||
return TypedName{_typedName.location, translateIdentifier(_typedName.name), _typedName.type};
|
return TypedName{_typedName.debugData, translateIdentifier(_typedName.name), _typedName.type};
|
||||||
}
|
}
|
||||||
|
|
||||||
YulString FunctionCopier::translateIdentifier(YulString _name)
|
YulString FunctionCopier::translateIdentifier(YulString _name)
|
||||||
|
@ -95,13 +95,13 @@ void CommonSubexpressionEliminator::visit(Expression& _e)
|
|||||||
|
|
||||||
if (Identifier const* identifier = get_if<Identifier>(&_e))
|
if (Identifier const* identifier = get_if<Identifier>(&_e))
|
||||||
{
|
{
|
||||||
YulString name = identifier->name;
|
YulString identifierName = identifier->name;
|
||||||
if (m_value.count(name))
|
if (m_value.count(identifierName))
|
||||||
{
|
{
|
||||||
assertThrow(m_value.at(name).value, OptimizerException, "");
|
assertThrow(m_value.at(identifierName).value, OptimizerException, "");
|
||||||
if (Identifier const* value = get_if<Identifier>(m_value.at(name).value))
|
if (Identifier const* value = get_if<Identifier>(m_value.at(identifierName).value))
|
||||||
if (inScope(value->name))
|
if (inScope(value->name))
|
||||||
_e = Identifier{locationOf(_e), value->name};
|
_e = Identifier{debugDataOf(_e), value->name};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -120,7 +120,7 @@ void CommonSubexpressionEliminator::visit(Expression& _e)
|
|||||||
continue;
|
continue;
|
||||||
if (SyntacticallyEqual{}(_e, *value.value) && inScope(variable))
|
if (SyntacticallyEqual{}(_e, *value.value) && inScope(variable))
|
||||||
{
|
{
|
||||||
_e = Identifier{locationOf(_e), variable};
|
_e = Identifier{debugDataOf(_e), variable};
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,8 +44,8 @@ void ConditionalSimplifier::operator()(Switch& _switch)
|
|||||||
(*this)(*_case.value);
|
(*this)(*_case.value);
|
||||||
_case.body.statements.insert(_case.body.statements.begin(),
|
_case.body.statements.insert(_case.body.statements.begin(),
|
||||||
Assignment{
|
Assignment{
|
||||||
_case.body.location,
|
_case.body.debugData,
|
||||||
{Identifier{_case.body.location, expr}},
|
{Identifier{_case.body.debugData, expr}},
|
||||||
make_unique<Expression>(*_case.value)
|
make_unique<Expression>(*_case.value)
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -72,12 +72,12 @@ void ConditionalSimplifier::operator()(Block& _block)
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
YulString condition = std::get<Identifier>(*_if.condition).name;
|
YulString condition = std::get<Identifier>(*_if.condition).name;
|
||||||
langutil::SourceLocation location = _if.location;
|
std::shared_ptr<DebugData const> debugData = _if.debugData;
|
||||||
return make_vector<Statement>(
|
return make_vector<Statement>(
|
||||||
std::move(_s),
|
std::move(_s),
|
||||||
Assignment{
|
Assignment{
|
||||||
location,
|
debugData,
|
||||||
{Identifier{location, condition}},
|
{Identifier{debugData, condition}},
|
||||||
make_unique<Expression>(m_dialect.zeroLiteralForType(m_dialect.boolType))
|
make_unique<Expression>(m_dialect.zeroLiteralForType(m_dialect.boolType))
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -38,14 +38,14 @@ namespace
|
|||||||
{
|
{
|
||||||
|
|
||||||
ExpressionStatement makeDiscardCall(
|
ExpressionStatement makeDiscardCall(
|
||||||
langutil::SourceLocation const& _location,
|
std::shared_ptr<DebugData const> const& _debugData,
|
||||||
BuiltinFunction const& _discardFunction,
|
BuiltinFunction const& _discardFunction,
|
||||||
Expression&& _expression
|
Expression&& _expression
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
return {_location, FunctionCall{
|
return {_debugData, FunctionCall{
|
||||||
_location,
|
_debugData,
|
||||||
Identifier{_location, _discardFunction.name},
|
Identifier{_debugData, _discardFunction.name},
|
||||||
{std::move(_expression)}
|
{std::move(_expression)}
|
||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
@ -126,7 +126,7 @@ void ControlFlowSimplifier::visit(Statement& _st)
|
|||||||
|
|
||||||
if (isTerminating && m_numContinueStatements == 0 && m_numBreakStatements == 0)
|
if (isTerminating && m_numContinueStatements == 0 && m_numBreakStatements == 0)
|
||||||
{
|
{
|
||||||
If replacement{forLoop.location, std::move(forLoop.condition), std::move(forLoop.body)};
|
If replacement{forLoop.debugData, std::move(forLoop.condition), std::move(forLoop.body)};
|
||||||
if (controlFlow == TerminationFinder::ControlFlow::Break)
|
if (controlFlow == TerminationFinder::ControlFlow::Break)
|
||||||
replacement.body.statements.resize(replacement.body.statements.size() - 1);
|
replacement.body.statements.resize(replacement.body.statements.size() - 1);
|
||||||
_st = std::move(replacement);
|
_st = std::move(replacement);
|
||||||
@ -149,7 +149,7 @@ void ControlFlowSimplifier::simplify(std::vector<yul::Statement>& _statements)
|
|||||||
{
|
{
|
||||||
OptionalStatements s = vector<Statement>{};
|
OptionalStatements s = vector<Statement>{};
|
||||||
s->emplace_back(makeDiscardCall(
|
s->emplace_back(makeDiscardCall(
|
||||||
_ifStmt.location,
|
_ifStmt.debugData,
|
||||||
*m_dialect.discardFunction(m_dialect.boolType),
|
*m_dialect.discardFunction(m_dialect.boolType),
|
||||||
std::move(*_ifStmt.condition)
|
std::move(*_ifStmt.condition)
|
||||||
));
|
));
|
||||||
@ -191,10 +191,8 @@ OptionalStatements ControlFlowSimplifier::reduceNoCaseSwitch(Switch& _switchStmt
|
|||||||
if (!discardFunction)
|
if (!discardFunction)
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
auto loc = locationOf(*_switchStmt.expression);
|
|
||||||
|
|
||||||
return make_vector<Statement>(makeDiscardCall(
|
return make_vector<Statement>(makeDiscardCall(
|
||||||
loc,
|
debugDataOf(*_switchStmt.expression),
|
||||||
*discardFunction,
|
*discardFunction,
|
||||||
std::move(*_switchStmt.expression)
|
std::move(*_switchStmt.expression)
|
||||||
));
|
));
|
||||||
@ -205,17 +203,17 @@ OptionalStatements ControlFlowSimplifier::reduceSingleCaseSwitch(Switch& _switch
|
|||||||
yulAssert(_switchStmt.cases.size() == 1, "Expected only one case!");
|
yulAssert(_switchStmt.cases.size() == 1, "Expected only one case!");
|
||||||
|
|
||||||
auto& switchCase = _switchStmt.cases.front();
|
auto& switchCase = _switchStmt.cases.front();
|
||||||
auto loc = locationOf(*_switchStmt.expression);
|
shared_ptr<DebugData const> debugData = debugDataOf(*_switchStmt.expression);
|
||||||
YulString type = m_typeInfo.typeOf(*_switchStmt.expression);
|
YulString type = m_typeInfo.typeOf(*_switchStmt.expression);
|
||||||
if (switchCase.value)
|
if (switchCase.value)
|
||||||
{
|
{
|
||||||
if (!m_dialect.equalityFunction(type))
|
if (!m_dialect.equalityFunction(type))
|
||||||
return {};
|
return {};
|
||||||
return make_vector<Statement>(If{
|
return make_vector<Statement>(If{
|
||||||
std::move(_switchStmt.location),
|
std::move(_switchStmt.debugData),
|
||||||
make_unique<Expression>(FunctionCall{
|
make_unique<Expression>(FunctionCall{
|
||||||
loc,
|
debugData,
|
||||||
Identifier{loc, m_dialect.equalityFunction(type)->name},
|
Identifier{debugData, m_dialect.equalityFunction(type)->name},
|
||||||
{std::move(*switchCase.value), std::move(*_switchStmt.expression)}
|
{std::move(*switchCase.value), std::move(*_switchStmt.expression)}
|
||||||
}),
|
}),
|
||||||
std::move(switchCase.body)
|
std::move(switchCase.body)
|
||||||
@ -228,7 +226,7 @@ OptionalStatements ControlFlowSimplifier::reduceSingleCaseSwitch(Switch& _switch
|
|||||||
|
|
||||||
return make_vector<Statement>(
|
return make_vector<Statement>(
|
||||||
makeDiscardCall(
|
makeDiscardCall(
|
||||||
loc,
|
debugData,
|
||||||
*m_dialect.discardFunction(type),
|
*m_dialect.discardFunction(type),
|
||||||
std::move(*_switchStmt.expression)
|
std::move(*_switchStmt.expression)
|
||||||
),
|
),
|
||||||
|
@ -39,5 +39,5 @@ void ExpressionSimplifier::visit(Expression& _expression)
|
|||||||
ASTModifier::visit(_expression);
|
ASTModifier::visit(_expression);
|
||||||
|
|
||||||
while (auto const* match = SimplificationRules::findFirstMatch(_expression, m_dialect, m_value))
|
while (auto const* match = SimplificationRules::findFirstMatch(_expression, m_dialect, m_value))
|
||||||
_expression = match->action().toExpression(locationOf(_expression));
|
_expression = match->action().toExpression(debugDataOf(_expression));
|
||||||
}
|
}
|
||||||
|
@ -101,15 +101,15 @@ void ExpressionSplitter::outlineExpression(Expression& _expr)
|
|||||||
|
|
||||||
visit(_expr);
|
visit(_expr);
|
||||||
|
|
||||||
SourceLocation location = locationOf(_expr);
|
shared_ptr<DebugData const> debugData = debugDataOf(_expr);
|
||||||
YulString var = m_nameDispenser.newName({});
|
YulString var = m_nameDispenser.newName({});
|
||||||
YulString type = m_typeInfo.typeOf(_expr);
|
YulString type = m_typeInfo.typeOf(_expr);
|
||||||
m_statementsToPrefix.emplace_back(VariableDeclaration{
|
m_statementsToPrefix.emplace_back(VariableDeclaration{
|
||||||
location,
|
debugData,
|
||||||
{{TypedName{location, var, type}}},
|
{{TypedName{debugData, var, type}}},
|
||||||
make_unique<Expression>(std::move(_expr))
|
make_unique<Expression>(std::move(_expr))
|
||||||
});
|
});
|
||||||
_expr = Identifier{location, var};
|
_expr = Identifier{debugData, var};
|
||||||
m_typeInfo.setVariableType(var, type);
|
m_typeInfo.setVariableType(var, type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,25 +39,25 @@ void ForLoopConditionIntoBody::operator()(ForLoop& _forLoop)
|
|||||||
!holds_alternative<Identifier>(*_forLoop.condition)
|
!holds_alternative<Identifier>(*_forLoop.condition)
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
langutil::SourceLocation const loc = locationOf(*_forLoop.condition);
|
shared_ptr<DebugData const> debugData = debugDataOf(*_forLoop.condition);
|
||||||
|
|
||||||
_forLoop.body.statements.emplace(
|
_forLoop.body.statements.emplace(
|
||||||
begin(_forLoop.body.statements),
|
begin(_forLoop.body.statements),
|
||||||
If {
|
If {
|
||||||
loc,
|
debugData,
|
||||||
make_unique<Expression>(
|
make_unique<Expression>(
|
||||||
FunctionCall {
|
FunctionCall {
|
||||||
loc,
|
debugData,
|
||||||
{loc, m_dialect.booleanNegationFunction()->name},
|
{debugData, m_dialect.booleanNegationFunction()->name},
|
||||||
util::make_vector<Expression>(std::move(*_forLoop.condition))
|
util::make_vector<Expression>(std::move(*_forLoop.condition))
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
Block {loc, util::make_vector<Statement>(Break{{}})}
|
Block {debugData, util::make_vector<Statement>(Break{{}})}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
_forLoop.condition = make_unique<Expression>(
|
_forLoop.condition = make_unique<Expression>(
|
||||||
Literal {
|
Literal {
|
||||||
loc,
|
debugData,
|
||||||
LiteralKind::Boolean,
|
LiteralKind::Boolean,
|
||||||
"true"_yulstring,
|
"true"_yulstring,
|
||||||
m_dialect.boolType
|
m_dialect.boolType
|
||||||
|
@ -55,7 +55,7 @@ void ForLoopConditionOutOfBody::operator()(ForLoop& _forLoop)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
YulString iszero = m_dialect.booleanNegationFunction()->name;
|
YulString iszero = m_dialect.booleanNegationFunction()->name;
|
||||||
langutil::SourceLocation location = locationOf(*firstStatement.condition);
|
shared_ptr<DebugData const> debugData = debugDataOf(*firstStatement.condition);
|
||||||
|
|
||||||
if (
|
if (
|
||||||
holds_alternative<FunctionCall>(*firstStatement.condition) &&
|
holds_alternative<FunctionCall>(*firstStatement.condition) &&
|
||||||
@ -64,8 +64,8 @@ void ForLoopConditionOutOfBody::operator()(ForLoop& _forLoop)
|
|||||||
_forLoop.condition = make_unique<Expression>(std::move(std::get<FunctionCall>(*firstStatement.condition).arguments.front()));
|
_forLoop.condition = make_unique<Expression>(std::move(std::get<FunctionCall>(*firstStatement.condition).arguments.front()));
|
||||||
else
|
else
|
||||||
_forLoop.condition = make_unique<Expression>(FunctionCall{
|
_forLoop.condition = make_unique<Expression>(FunctionCall{
|
||||||
location,
|
debugData,
|
||||||
Identifier{location, iszero},
|
Identifier{debugData, iszero},
|
||||||
util::make_vector<Expression>(
|
util::make_vector<Expression>(
|
||||||
std::move(*firstStatement.condition)
|
std::move(*firstStatement.condition)
|
||||||
)
|
)
|
||||||
|
@ -268,7 +268,7 @@ vector<Statement> InlineModifier::performInline(Statement& _statement, FunctionC
|
|||||||
auto newVariable = [&](TypedName const& _existingVariable, Expression* _value) {
|
auto newVariable = [&](TypedName const& _existingVariable, Expression* _value) {
|
||||||
YulString newName = m_nameDispenser.newName(_existingVariable.name);
|
YulString newName = m_nameDispenser.newName(_existingVariable.name);
|
||||||
variableReplacements[_existingVariable.name] = newName;
|
variableReplacements[_existingVariable.name] = newName;
|
||||||
VariableDeclaration varDecl{_funCall.location, {{_funCall.location, newName, _existingVariable.type}}, {}};
|
VariableDeclaration varDecl{_funCall.debugData, {{_funCall.debugData, newName, _existingVariable.type}}, {}};
|
||||||
if (_value)
|
if (_value)
|
||||||
varDecl.value = make_unique<Expression>(std::move(*_value));
|
varDecl.value = make_unique<Expression>(std::move(*_value));
|
||||||
else
|
else
|
||||||
@ -290,10 +290,10 @@ vector<Statement> InlineModifier::performInline(Statement& _statement, FunctionC
|
|||||||
{
|
{
|
||||||
for (size_t i = 0; i < _assignment.variableNames.size(); ++i)
|
for (size_t i = 0; i < _assignment.variableNames.size(); ++i)
|
||||||
newStatements.emplace_back(Assignment{
|
newStatements.emplace_back(Assignment{
|
||||||
_assignment.location,
|
_assignment.debugData,
|
||||||
{_assignment.variableNames[i]},
|
{_assignment.variableNames[i]},
|
||||||
make_unique<Expression>(Identifier{
|
make_unique<Expression>(Identifier{
|
||||||
_assignment.location,
|
_assignment.debugData,
|
||||||
variableReplacements.at(function->returnVariables[i].name)
|
variableReplacements.at(function->returnVariables[i].name)
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
@ -302,10 +302,10 @@ vector<Statement> InlineModifier::performInline(Statement& _statement, FunctionC
|
|||||||
{
|
{
|
||||||
for (size_t i = 0; i < _varDecl.variables.size(); ++i)
|
for (size_t i = 0; i < _varDecl.variables.size(); ++i)
|
||||||
newStatements.emplace_back(VariableDeclaration{
|
newStatements.emplace_back(VariableDeclaration{
|
||||||
_varDecl.location,
|
_varDecl.debugData,
|
||||||
{std::move(_varDecl.variables[i])},
|
{std::move(_varDecl.variables[i])},
|
||||||
make_unique<Expression>(Identifier{
|
make_unique<Expression>(Identifier{
|
||||||
_varDecl.location,
|
_varDecl.debugData,
|
||||||
variableReplacements.at(function->returnVariables[i].name)
|
variableReplacements.at(function->returnVariables[i].name)
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
@ -35,7 +35,7 @@ void FunctionGrouper::operator()(Block& _block)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
vector<Statement> reordered;
|
vector<Statement> reordered;
|
||||||
reordered.emplace_back(Block{_block.location, {}});
|
reordered.emplace_back(Block{_block.debugData, {}});
|
||||||
|
|
||||||
for (auto&& statement: _block.statements)
|
for (auto&& statement: _block.statements)
|
||||||
{
|
{
|
||||||
|
@ -41,7 +41,7 @@ void FunctionHoister::operator()(Block& _block)
|
|||||||
if (holds_alternative<FunctionDefinition>(statement))
|
if (holds_alternative<FunctionDefinition>(statement))
|
||||||
{
|
{
|
||||||
m_functions.emplace_back(std::move(statement));
|
m_functions.emplace_back(std::move(statement));
|
||||||
statement = Block{_block.location, {}};
|
statement = Block{_block.debugData, {}};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
removeEmptyBlocks(_block);
|
removeEmptyBlocks(_block);
|
||||||
|
@ -104,7 +104,7 @@ FunctionDefinition FunctionSpecializer::specialize(
|
|||||||
if (argument)
|
if (argument)
|
||||||
missingVariableDeclarations.emplace_back(
|
missingVariableDeclarations.emplace_back(
|
||||||
VariableDeclaration{
|
VariableDeclaration{
|
||||||
_f.location,
|
_f.debugData,
|
||||||
vector<TypedName>{newFunction.parameters[index]},
|
vector<TypedName>{newFunction.parameters[index]},
|
||||||
make_unique<Expression>(move(*argument))
|
make_unique<Expression>(move(*argument))
|
||||||
}
|
}
|
||||||
|
@ -84,7 +84,7 @@ Expression KnowledgeBase::simplify(Expression _expression)
|
|||||||
arg = simplify(arg);
|
arg = simplify(arg);
|
||||||
|
|
||||||
if (auto match = SimplificationRules::findFirstMatch(_expression, m_dialect, m_variableValues))
|
if (auto match = SimplificationRules::findFirstMatch(_expression, m_dialect, m_variableValues))
|
||||||
return simplify(match->action().toExpression(locationOf(_expression)));
|
return simplify(match->action().toExpression(debugDataOf(_expression)));
|
||||||
|
|
||||||
return _expression;
|
return _expression;
|
||||||
}
|
}
|
||||||
|
@ -85,12 +85,12 @@ void LoadResolver::tryResolve(
|
|||||||
{
|
{
|
||||||
if (auto value = util::valueOrNullptr(m_storage, key))
|
if (auto value = util::valueOrNullptr(m_storage, key))
|
||||||
if (inScope(*value))
|
if (inScope(*value))
|
||||||
_e = Identifier{locationOf(_e), *value};
|
_e = Identifier{debugDataOf(_e), *value};
|
||||||
}
|
}
|
||||||
else if (!m_containsMSize && _location == StoreLoadLocation::Memory)
|
else if (!m_containsMSize && _location == StoreLoadLocation::Memory)
|
||||||
if (auto value = util::valueOrNullptr(m_memory, key))
|
if (auto value = util::valueOrNullptr(m_memory, key))
|
||||||
if (inScope(*value))
|
if (inScope(*value))
|
||||||
_e = Identifier{locationOf(_e), *value};
|
_e = Identifier{debugDataOf(_e), *value};
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoadResolver::tryEvaluateKeccak(
|
void LoadResolver::tryEvaluateKeccak(
|
||||||
@ -140,7 +140,7 @@ void LoadResolver::tryEvaluateKeccak(
|
|||||||
bytes contentAsBytes = toBigEndian(*memoryContent);
|
bytes contentAsBytes = toBigEndian(*memoryContent);
|
||||||
contentAsBytes.resize(static_cast<size_t>(*byteLength));
|
contentAsBytes.resize(static_cast<size_t>(*byteLength));
|
||||||
_e = Literal{
|
_e = Literal{
|
||||||
locationOf(_e),
|
debugDataOf(_e),
|
||||||
LiteralKind::Number,
|
LiteralKind::Number,
|
||||||
YulString{u256(keccak256(contentAsBytes)).str()},
|
YulString{u256(keccak256(contentAsBytes)).str()},
|
||||||
m_dialect.defaultType
|
m_dialect.defaultType
|
||||||
|
@ -44,7 +44,7 @@ void MainFunction::operator()(Block& _block)
|
|||||||
|
|
||||||
Block& block = std::get<Block>(_block.statements[0]);
|
Block& block = std::get<Block>(_block.statements[0]);
|
||||||
FunctionDefinition main{
|
FunctionDefinition main{
|
||||||
block.location,
|
block.debugData,
|
||||||
"main"_yulstring,
|
"main"_yulstring,
|
||||||
{},
|
{},
|
||||||
{},
|
{},
|
||||||
|
@ -79,7 +79,7 @@ void ReasoningBasedSimplifier::operator()(If& _if)
|
|||||||
if (result == CheckResult::UNSATISFIABLE)
|
if (result == CheckResult::UNSATISFIABLE)
|
||||||
{
|
{
|
||||||
Literal trueCondition = m_dialect.trueLiteral();
|
Literal trueCondition = m_dialect.trueLiteral();
|
||||||
trueCondition.location = locationOf(*_if.condition);
|
trueCondition.debugData = debugDataOf(*_if.condition);
|
||||||
_if.condition = make_unique<yul::Expression>(move(trueCondition));
|
_if.condition = make_unique<yul::Expression>(move(trueCondition));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -91,7 +91,7 @@ void ReasoningBasedSimplifier::operator()(If& _if)
|
|||||||
if (result2 == CheckResult::UNSATISFIABLE)
|
if (result2 == CheckResult::UNSATISFIABLE)
|
||||||
{
|
{
|
||||||
Literal falseCondition = m_dialect.zeroLiteralForType(m_dialect.boolType);
|
Literal falseCondition = m_dialect.zeroLiteralForType(m_dialect.boolType);
|
||||||
falseCondition.location = locationOf(*_if.condition);
|
falseCondition.debugData = debugDataOf(*_if.condition);
|
||||||
_if.condition = make_unique<yul::Expression>(move(falseCondition));
|
_if.condition = make_unique<yul::Expression>(move(falseCondition));
|
||||||
_if.body = yul::Block{};
|
_if.body = yul::Block{};
|
||||||
// Nothing left to be done.
|
// Nothing left to be done.
|
||||||
|
@ -66,12 +66,12 @@ void SSAReverser::operator()(Block& _block)
|
|||||||
else
|
else
|
||||||
return util::make_vector<Statement>(
|
return util::make_vector<Statement>(
|
||||||
Assignment{
|
Assignment{
|
||||||
std::move(assignment->location),
|
std::move(assignment->debugData),
|
||||||
assignment->variableNames,
|
assignment->variableNames,
|
||||||
std::move(varDecl->value)
|
std::move(varDecl->value)
|
||||||
},
|
},
|
||||||
VariableDeclaration{
|
VariableDeclaration{
|
||||||
std::move(varDecl->location),
|
std::move(varDecl->debugData),
|
||||||
std::move(varDecl->variables),
|
std::move(varDecl->variables),
|
||||||
std::make_unique<Expression>(assignment->variableNames.front())
|
std::make_unique<Expression>(assignment->variableNames.front())
|
||||||
}
|
}
|
||||||
@ -97,17 +97,17 @@ void SSAReverser::operator()(Block& _block)
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
auto varIdentifier2 = std::make_unique<Expression>(Identifier{
|
auto varIdentifier2 = std::make_unique<Expression>(Identifier{
|
||||||
varDecl2->variables.front().location,
|
varDecl2->variables.front().debugData,
|
||||||
varDecl2->variables.front().name
|
varDecl2->variables.front().name
|
||||||
});
|
});
|
||||||
return util::make_vector<Statement>(
|
return util::make_vector<Statement>(
|
||||||
VariableDeclaration{
|
VariableDeclaration{
|
||||||
std::move(varDecl2->location),
|
std::move(varDecl2->debugData),
|
||||||
std::move(varDecl2->variables),
|
std::move(varDecl2->variables),
|
||||||
std::move(varDecl->value)
|
std::move(varDecl->value)
|
||||||
},
|
},
|
||||||
VariableDeclaration{
|
VariableDeclaration{
|
||||||
std::move(varDecl->location),
|
std::move(varDecl->debugData),
|
||||||
std::move(varDecl->variables),
|
std::move(varDecl->variables),
|
||||||
std::move(varIdentifier2)
|
std::move(varIdentifier2)
|
||||||
}
|
}
|
||||||
|
@ -85,19 +85,19 @@ void IntroduceSSA::operator()(Block& _block)
|
|||||||
|
|
||||||
// Replace "let a := v" by "let a_1 := v let a := a_1"
|
// Replace "let a := v" by "let a_1 := v let a := a_1"
|
||||||
// Replace "let a, b := v" by "let a_1, b_1 := v let a := a_1 let b := b_2"
|
// Replace "let a, b := v" by "let a_1, b_1 := v let a := a_1 let b := b_2"
|
||||||
auto loc = varDecl.location;
|
shared_ptr<DebugData const> debugData = varDecl.debugData;
|
||||||
vector<Statement> statements;
|
vector<Statement> statements;
|
||||||
statements.emplace_back(VariableDeclaration{loc, {}, std::move(varDecl.value)});
|
statements.emplace_back(VariableDeclaration{debugData, {}, std::move(varDecl.value)});
|
||||||
TypedNameList newVariables;
|
TypedNameList newVariables;
|
||||||
for (auto const& var: varDecl.variables)
|
for (auto const& var: varDecl.variables)
|
||||||
{
|
{
|
||||||
YulString oldName = var.name;
|
YulString oldName = var.name;
|
||||||
YulString newName = m_nameDispenser.newName(oldName);
|
YulString newName = m_nameDispenser.newName(oldName);
|
||||||
newVariables.emplace_back(TypedName{loc, newName, var.type});
|
newVariables.emplace_back(TypedName{debugData, newName, var.type});
|
||||||
statements.emplace_back(VariableDeclaration{
|
statements.emplace_back(VariableDeclaration{
|
||||||
loc,
|
debugData,
|
||||||
{TypedName{loc, oldName, var.type}},
|
{TypedName{debugData, oldName, var.type}},
|
||||||
make_unique<Expression>(Identifier{loc, newName})
|
make_unique<Expression>(Identifier{debugData, newName})
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
||||||
@ -112,23 +112,22 @@ void IntroduceSSA::operator()(Block& _block)
|
|||||||
|
|
||||||
// Replace "a := v" by "let a_1 := v a := v"
|
// Replace "a := v" by "let a_1 := v a := v"
|
||||||
// Replace "a, b := v" by "let a_1, b_1 := v a := a_1 b := b_2"
|
// Replace "a, b := v" by "let a_1, b_1 := v a := a_1 b := b_2"
|
||||||
auto loc = assignment.location;
|
std::shared_ptr<DebugData const> debugData = assignment.debugData;
|
||||||
vector<Statement> statements;
|
vector<Statement> statements;
|
||||||
statements.emplace_back(VariableDeclaration{loc, {}, std::move(assignment.value)});
|
statements.emplace_back(VariableDeclaration{debugData, {}, std::move(assignment.value)});
|
||||||
TypedNameList newVariables;
|
TypedNameList newVariables;
|
||||||
for (auto const& var: assignment.variableNames)
|
for (auto const& var: assignment.variableNames)
|
||||||
{
|
{
|
||||||
YulString oldName = var.name;
|
YulString oldName = var.name;
|
||||||
YulString newName = m_nameDispenser.newName(oldName);
|
YulString newName = m_nameDispenser.newName(oldName);
|
||||||
newVariables.emplace_back(TypedName{
|
newVariables.emplace_back(TypedName{debugData,
|
||||||
loc,
|
|
||||||
newName,
|
newName,
|
||||||
m_typeInfo.typeOfVariable(oldName)
|
m_typeInfo.typeOfVariable(oldName)
|
||||||
});
|
});
|
||||||
statements.emplace_back(Assignment{
|
statements.emplace_back(Assignment{
|
||||||
loc,
|
debugData,
|
||||||
{Identifier{loc, oldName}},
|
{Identifier{debugData, oldName}},
|
||||||
make_unique<Expression>(Identifier{loc, newName})
|
make_unique<Expression>(Identifier{debugData, newName})
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
||||||
@ -238,9 +237,9 @@ void IntroduceControlFlowSSA::operator()(Block& _block)
|
|||||||
{
|
{
|
||||||
YulString newName = m_nameDispenser.newName(toReassign);
|
YulString newName = m_nameDispenser.newName(toReassign);
|
||||||
toPrepend.emplace_back(VariableDeclaration{
|
toPrepend.emplace_back(VariableDeclaration{
|
||||||
locationOf(_s),
|
debugDataOf(_s),
|
||||||
{TypedName{locationOf(_s), newName, m_typeInfo.typeOfVariable(toReassign)}},
|
{TypedName{debugDataOf(_s), newName, m_typeInfo.typeOfVariable(toReassign)}},
|
||||||
make_unique<Expression>(Identifier{locationOf(_s), toReassign})
|
make_unique<Expression>(Identifier{debugDataOf(_s), toReassign})
|
||||||
});
|
});
|
||||||
assignedVariables.insert(toReassign);
|
assignedVariables.insert(toReassign);
|
||||||
}
|
}
|
||||||
|
@ -234,27 +234,26 @@ evmasm::Instruction Pattern::instruction() const
|
|||||||
return m_instruction;
|
return m_instruction;
|
||||||
}
|
}
|
||||||
|
|
||||||
Expression Pattern::toExpression(SourceLocation const& _location) const
|
Expression Pattern::toExpression(shared_ptr<DebugData const> const& _debugData) const
|
||||||
{
|
{
|
||||||
if (matchGroup())
|
if (matchGroup())
|
||||||
return ASTCopier().translate(matchGroupValue());
|
return ASTCopier().translate(matchGroupValue());
|
||||||
if (m_kind == PatternKind::Constant)
|
if (m_kind == PatternKind::Constant)
|
||||||
{
|
{
|
||||||
assertThrow(m_data, OptimizerException, "No match group and no constant value given.");
|
assertThrow(m_data, OptimizerException, "No match group and no constant value given.");
|
||||||
return Literal{_location, LiteralKind::Number, YulString{util::formatNumber(*m_data)}, {}};
|
return Literal{_debugData, LiteralKind::Number, YulString{util::formatNumber(*m_data)}, {}};
|
||||||
}
|
}
|
||||||
else if (m_kind == PatternKind::Operation)
|
else if (m_kind == PatternKind::Operation)
|
||||||
{
|
{
|
||||||
vector<Expression> arguments;
|
vector<Expression> arguments;
|
||||||
for (auto const& arg: m_arguments)
|
for (auto const& arg: m_arguments)
|
||||||
arguments.emplace_back(arg.toExpression(_location));
|
arguments.emplace_back(arg.toExpression(_debugData));
|
||||||
|
|
||||||
string name = instructionInfo(m_instruction).name;
|
string name = instructionInfo(m_instruction).name;
|
||||||
transform(begin(name), end(name), begin(name), [](auto _c) { return tolower(_c); });
|
transform(begin(name), end(name), begin(name), [](auto _c) { return tolower(_c); });
|
||||||
|
|
||||||
return FunctionCall{
|
return FunctionCall{_debugData,
|
||||||
_location,
|
Identifier{_debugData, YulString{name}},
|
||||||
Identifier{_location, YulString{name}},
|
|
||||||
std::move(arguments)
|
std::move(arguments)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -130,7 +130,7 @@ public:
|
|||||||
|
|
||||||
/// Turns this pattern into an actual expression. Should only be called
|
/// Turns this pattern into an actual expression. Should only be called
|
||||||
/// for patterns resulting from an action, i.e. with match groups assigned.
|
/// for patterns resulting from an action, i.e. with match groups assigned.
|
||||||
Expression toExpression(langutil::SourceLocation const& _location) const;
|
Expression toExpression(std::shared_ptr<DebugData const> const& _debugData) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Expression const& matchGroupValue() const;
|
Expression const& matchGroupValue() const;
|
||||||
|
@ -31,7 +31,7 @@ namespace
|
|||||||
{
|
{
|
||||||
vector<Statement> generateMemoryStore(
|
vector<Statement> generateMemoryStore(
|
||||||
Dialect const& _dialect,
|
Dialect const& _dialect,
|
||||||
langutil::SourceLocation const& _loc,
|
shared_ptr<DebugData const> const& _debugData,
|
||||||
YulString _mpos,
|
YulString _mpos,
|
||||||
Expression _value
|
Expression _value
|
||||||
)
|
)
|
||||||
@ -39,26 +39,26 @@ vector<Statement> generateMemoryStore(
|
|||||||
BuiltinFunction const* memoryStoreFunction = _dialect.memoryStoreFunction(_dialect.defaultType);
|
BuiltinFunction const* memoryStoreFunction = _dialect.memoryStoreFunction(_dialect.defaultType);
|
||||||
yulAssert(memoryStoreFunction, "");
|
yulAssert(memoryStoreFunction, "");
|
||||||
vector<Statement> result;
|
vector<Statement> result;
|
||||||
result.emplace_back(ExpressionStatement{_loc, FunctionCall{
|
result.emplace_back(ExpressionStatement{_debugData, FunctionCall{
|
||||||
_loc,
|
_debugData,
|
||||||
Identifier{_loc, memoryStoreFunction->name},
|
Identifier{_debugData, memoryStoreFunction->name},
|
||||||
{
|
{
|
||||||
Literal{_loc, LiteralKind::Number, _mpos, {}},
|
Literal{_debugData, LiteralKind::Number, _mpos, {}},
|
||||||
std::move(_value)
|
std::move(_value)
|
||||||
}
|
}
|
||||||
}});
|
}});
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
FunctionCall generateMemoryLoad(Dialect const& _dialect, langutil::SourceLocation const& _loc, YulString _mpos)
|
FunctionCall generateMemoryLoad(Dialect const& _dialect, std::shared_ptr<DebugData const> const& _debugData, YulString _mpos)
|
||||||
{
|
{
|
||||||
BuiltinFunction const* memoryLoadFunction = _dialect.memoryLoadFunction(_dialect.defaultType);
|
BuiltinFunction const* memoryLoadFunction = _dialect.memoryLoadFunction(_dialect.defaultType);
|
||||||
yulAssert(memoryLoadFunction, "");
|
yulAssert(memoryLoadFunction, "");
|
||||||
return FunctionCall{
|
return FunctionCall{
|
||||||
_loc,
|
_debugData,
|
||||||
Identifier{_loc, memoryLoadFunction->name}, {
|
Identifier{_debugData, memoryLoadFunction->name}, {
|
||||||
Literal{
|
Literal{
|
||||||
_loc,
|
_debugData,
|
||||||
LiteralKind::Number,
|
LiteralKind::Number,
|
||||||
_mpos,
|
_mpos,
|
||||||
{}
|
{}
|
||||||
@ -123,39 +123,38 @@ void StackToMemoryMover::operator()(Block& _block)
|
|||||||
if (!leftHandSideNeedsMoving)
|
if (!leftHandSideNeedsMoving)
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
langutil::SourceLocation loc = _stmt.location;
|
|
||||||
|
|
||||||
if (_variables.size() == 1)
|
if (_variables.size() == 1)
|
||||||
{
|
{
|
||||||
optional<YulString> offset = m_memoryOffsetTracker(_variables.front().name);
|
optional<YulString> offset = m_memoryOffsetTracker(_variables.front().name);
|
||||||
yulAssert(offset, "");
|
yulAssert(offset, "");
|
||||||
return generateMemoryStore(
|
return generateMemoryStore(
|
||||||
m_context.dialect,
|
m_context.dialect,
|
||||||
loc,
|
_stmt.debugData,
|
||||||
*offset,
|
*offset,
|
||||||
_stmt.value ? *std::move(_stmt.value) : Literal{loc, LiteralKind::Number, "0"_yulstring, {}}
|
_stmt.value ? *std::move(_stmt.value) : Literal{_stmt.debugData, LiteralKind::Number, "0"_yulstring, {}}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
VariableDeclaration tempDecl{loc, {}, std::move(_stmt.value)};
|
VariableDeclaration tempDecl{_stmt.debugData, {}, std::move(_stmt.value)};
|
||||||
vector<Statement> memoryAssignments;
|
vector<Statement> memoryAssignments;
|
||||||
vector<Statement> variableAssignments;
|
vector<Statement> variableAssignments;
|
||||||
for (auto& var: _variables)
|
for (auto& var: _variables)
|
||||||
{
|
{
|
||||||
YulString tempVarName = m_nameDispenser.newName(var.name);
|
YulString tempVarName = m_nameDispenser.newName(var.name);
|
||||||
tempDecl.variables.emplace_back(TypedName{var.location, tempVarName, {}});
|
tempDecl.variables.emplace_back(TypedName{var.debugData, tempVarName, {}});
|
||||||
|
|
||||||
if (optional<YulString> offset = m_memoryOffsetTracker(var.name))
|
if (optional<YulString> offset = m_memoryOffsetTracker(var.name))
|
||||||
memoryAssignments += generateMemoryStore(
|
memoryAssignments += generateMemoryStore(
|
||||||
m_context.dialect,
|
m_context.dialect,
|
||||||
loc,
|
_stmt.debugData,
|
||||||
*offset,
|
*offset,
|
||||||
Identifier{loc, tempVarName}
|
Identifier{_stmt.debugData, tempVarName}
|
||||||
);
|
);
|
||||||
else
|
else
|
||||||
variableAssignments.emplace_back(StatementType{
|
variableAssignments.emplace_back(StatementType{
|
||||||
loc, {move(var)},
|
_stmt.debugData,
|
||||||
make_unique<Expression>(Identifier{loc, tempVarName})
|
{move(var)},
|
||||||
|
make_unique<Expression>(Identifier{_stmt.debugData, tempVarName})
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
std::vector<Statement> result;
|
std::vector<Statement> result;
|
||||||
@ -191,7 +190,7 @@ void StackToMemoryMover::visit(Expression& _expression)
|
|||||||
ASTModifier::visit(_expression);
|
ASTModifier::visit(_expression);
|
||||||
if (Identifier* identifier = std::get_if<Identifier>(&_expression))
|
if (Identifier* identifier = std::get_if<Identifier>(&_expression))
|
||||||
if (optional<YulString> offset = m_memoryOffsetTracker(identifier->name))
|
if (optional<YulString> offset = m_memoryOffsetTracker(identifier->name))
|
||||||
_expression = generateMemoryLoad(m_context.dialect, identifier->location, *offset);
|
_expression = generateMemoryLoad(m_context.dialect, identifier->debugData, *offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
optional<YulString> StackToMemoryMover::VariableMemoryOffsetTracker::operator()(YulString _variable) const
|
optional<YulString> StackToMemoryMover::VariableMemoryOffsetTracker::operator()(YulString _variable) const
|
||||||
|
@ -40,33 +40,31 @@ FunctionDefinition unusedFunctionsCommon::createLinkingFunction(
|
|||||||
auto generateTypedName = [&](TypedName t)
|
auto generateTypedName = [&](TypedName t)
|
||||||
{
|
{
|
||||||
return TypedName{
|
return TypedName{
|
||||||
t.location,
|
t.debugData,
|
||||||
_nameDispenser.newName(t.name),
|
_nameDispenser.newName(t.name),
|
||||||
t.type
|
t.type
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
langutil::SourceLocation loc = _original.location;
|
|
||||||
|
|
||||||
FunctionDefinition linkingFunction{
|
FunctionDefinition linkingFunction{
|
||||||
loc,
|
_original.debugData,
|
||||||
_linkingFunctionName,
|
_linkingFunctionName,
|
||||||
util::applyMap(_original.parameters, generateTypedName),
|
util::applyMap(_original.parameters, generateTypedName),
|
||||||
util::applyMap(_original.returnVariables, generateTypedName),
|
util::applyMap(_original.returnVariables, generateTypedName),
|
||||||
{loc, {}} // body
|
{_original.debugData, {}} // body
|
||||||
};
|
};
|
||||||
|
|
||||||
FunctionCall call{loc, Identifier{loc, _originalFunctionName}, {}};
|
FunctionCall call{_original.debugData, Identifier{_original.debugData, _originalFunctionName}, {}};
|
||||||
for (auto const& p: filter(linkingFunction.parameters, _usedParametersAndReturns.first))
|
for (auto const& p: filter(linkingFunction.parameters, _usedParametersAndReturns.first))
|
||||||
call.arguments.emplace_back(Identifier{loc, p.name});
|
call.arguments.emplace_back(Identifier{_original.debugData, p.name});
|
||||||
|
|
||||||
Assignment assignment{loc, {}, nullptr};
|
Assignment assignment{_original.debugData, {}, nullptr};
|
||||||
|
|
||||||
for (auto const& r: filter(linkingFunction.returnVariables, _usedParametersAndReturns.second))
|
for (auto const& r: filter(linkingFunction.returnVariables, _usedParametersAndReturns.second))
|
||||||
assignment.variableNames.emplace_back(Identifier{loc, r.name});
|
assignment.variableNames.emplace_back(Identifier{_original.debugData, r.name});
|
||||||
|
|
||||||
if (assignment.variableNames.empty())
|
if (assignment.variableNames.empty())
|
||||||
linkingFunction.body.statements.emplace_back(ExpressionStatement{loc, std::move(call)});
|
linkingFunction.body.statements.emplace_back(ExpressionStatement{_original.debugData, std::move(call)});
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
assignment.value = std::make_unique<Expression>(std::move(call));
|
assignment.value = std::make_unique<Expression>(std::move(call));
|
||||||
|
@ -72,7 +72,7 @@ void UnusedPruner::operator()(Block& _block)
|
|||||||
if (!used(funDef.name))
|
if (!used(funDef.name))
|
||||||
{
|
{
|
||||||
subtractReferences(ReferencesCounter::countReferences(funDef.body));
|
subtractReferences(ReferencesCounter::countReferences(funDef.body));
|
||||||
statement = Block{std::move(funDef.location), {}};
|
statement = Block{std::move(funDef.debugData), {}};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (holds_alternative<VariableDeclaration>(statement))
|
else if (holds_alternative<VariableDeclaration>(statement))
|
||||||
@ -90,19 +90,19 @@ void UnusedPruner::operator()(Block& _block)
|
|||||||
))
|
))
|
||||||
{
|
{
|
||||||
if (!varDecl.value)
|
if (!varDecl.value)
|
||||||
statement = Block{std::move(varDecl.location), {}};
|
statement = Block{std::move(varDecl.debugData), {}};
|
||||||
else if (
|
else if (
|
||||||
SideEffectsCollector(m_dialect, *varDecl.value, m_functionSideEffects).
|
SideEffectsCollector(m_dialect, *varDecl.value, m_functionSideEffects).
|
||||||
canBeRemoved(m_allowMSizeOptimization)
|
canBeRemoved(m_allowMSizeOptimization)
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
subtractReferences(ReferencesCounter::countReferences(*varDecl.value));
|
subtractReferences(ReferencesCounter::countReferences(*varDecl.value));
|
||||||
statement = Block{std::move(varDecl.location), {}};
|
statement = Block{std::move(varDecl.debugData), {}};
|
||||||
}
|
}
|
||||||
else if (varDecl.variables.size() == 1 && m_dialect.discardFunction(varDecl.variables.front().type))
|
else if (varDecl.variables.size() == 1 && m_dialect.discardFunction(varDecl.variables.front().type))
|
||||||
statement = ExpressionStatement{varDecl.location, FunctionCall{
|
statement = ExpressionStatement{varDecl.debugData, FunctionCall{
|
||||||
varDecl.location,
|
varDecl.debugData,
|
||||||
{varDecl.location, m_dialect.discardFunction(varDecl.variables.front().type)->name},
|
{varDecl.debugData, m_dialect.discardFunction(varDecl.variables.front().type)->name},
|
||||||
{*std::move(varDecl.value)}
|
{*std::move(varDecl.value)}
|
||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
@ -116,7 +116,7 @@ void UnusedPruner::operator()(Block& _block)
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
subtractReferences(ReferencesCounter::countReferences(exprStmt.expression));
|
subtractReferences(ReferencesCounter::countReferences(exprStmt.expression));
|
||||||
statement = Block{std::move(exprStmt.location), {}};
|
statement = Block{std::move(exprStmt.debugData), {}};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -47,11 +47,10 @@ void VarDeclInitializer::operator()(Block& _block)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
OptionalStatements ret{vector<Statement>{}};
|
OptionalStatements ret{vector<Statement>{}};
|
||||||
langutil::SourceLocation loc{std::move(_varDecl.location)};
|
|
||||||
for (auto& var: _varDecl.variables)
|
for (auto& var: _varDecl.variables)
|
||||||
{
|
{
|
||||||
unique_ptr<Expression> expr = make_unique<Expression >(m_dialect.zeroLiteralForType(var.type));
|
unique_ptr<Expression> expr = make_unique<Expression >(m_dialect.zeroLiteralForType(var.type));
|
||||||
ret->emplace_back(VariableDeclaration{loc, {std::move(var)}, std::move(expr)});
|
ret->emplace_back(VariableDeclaration{std::move(_varDecl.debugData), {std::move(var)}, std::move(expr)});
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,6 @@
|
|||||||
#include <liblangutil/SourceReferenceFormatter.h>
|
#include <liblangutil/SourceReferenceFormatter.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/Disambiguator.h>
|
#include <libyul/optimiser/Disambiguator.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
|
@ -25,7 +25,6 @@
|
|||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/backends/wasm/WasmDialect.h>
|
#include <libyul/backends/wasm/WasmDialect.h>
|
||||||
#include <libyul/backends/wasm/EVMToEwasmTranslator.h>
|
#include <libyul/backends/wasm/EVMToEwasmTranslator.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AST.h>
|
#include <libyul/AST.h>
|
||||||
|
@ -24,11 +24,11 @@
|
|||||||
#include <test/libsolidity/ErrorCheck.h>
|
#include <test/libsolidity/ErrorCheck.h>
|
||||||
#include <test/libyul/Common.h>
|
#include <test/libyul/Common.h>
|
||||||
|
|
||||||
|
#include <libyul/AST.h>
|
||||||
#include <libyul/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libyul/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AST.h>
|
|
||||||
#include <libyul/Dialect.h>
|
#include <libyul/Dialect.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
@ -18,7 +18,6 @@
|
|||||||
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
|
|
||||||
#include <liblangutil/EVMVersion.h>
|
#include <liblangutil/EVMVersion.h>
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
|
@ -23,7 +23,6 @@
|
|||||||
#include <test/Common.h>
|
#include <test/Common.h>
|
||||||
|
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
|
||||||
|
@ -17,14 +17,12 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0
|
// SPDX-License-Identifier: GPL-3.0
|
||||||
|
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/Dialect.h>
|
#include <libyul/Dialect.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
|
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
|
||||||
#include <liblangutil/EVMVersion.h>
|
#include <liblangutil/EVMVersion.h>
|
||||||
|
|
||||||
#include <libsolutil/CommonIO.h>
|
#include <libsolutil/CommonIO.h>
|
||||||
|
@ -22,14 +22,12 @@
|
|||||||
#include <test/tools/yulInterpreter/Interpreter.h>
|
#include <test/tools/yulInterpreter/Interpreter.h>
|
||||||
|
|
||||||
#include <libyul/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/AsmParser.h>
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libyul/Dialect.h>
|
#include <libyul/Dialect.h>
|
||||||
#include <libyul/backends/evm/EVMDialect.h>
|
#include <libyul/backends/evm/EVMDialect.h>
|
||||||
#include <libyul/AssemblyStack.h>
|
#include <libyul/AssemblyStack.h>
|
||||||
|
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
|
||||||
#include <liblangutil/EVMVersion.h>
|
#include <liblangutil/EVMVersion.h>
|
||||||
#include <liblangutil/SourceReferenceFormatter.h>
|
#include <liblangutil/SourceReferenceFormatter.h>
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user