mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #5491 from ethereum/libyul-as-library
Libyul as static library (version 3 ;( )
This commit is contained in:
commit
9a94e503da
@ -47,6 +47,7 @@ configure_project(TESTS)
|
|||||||
add_subdirectory(libdevcore)
|
add_subdirectory(libdevcore)
|
||||||
add_subdirectory(liblangutil)
|
add_subdirectory(liblangutil)
|
||||||
add_subdirectory(libevmasm)
|
add_subdirectory(libevmasm)
|
||||||
|
add_subdirectory(libyul)
|
||||||
add_subdirectory(libsolidity)
|
add_subdirectory(libsolidity)
|
||||||
add_subdirectory(libsolc)
|
add_subdirectory(libsolc)
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Until we have a clear separation, libyul has to be included here
|
# Until we have a clear separation, libyul has to be included here
|
||||||
file(GLOB_RECURSE sources "*.cpp" "../libyul/*.cpp")
|
file(GLOB_RECURSE sources "*.cpp")
|
||||||
file(GLOB_RECURSE headers "*.h" "../libyul/*.h")
|
file(GLOB_RECURSE headers "*.h")
|
||||||
|
|
||||||
find_package(Z3 QUIET)
|
find_package(Z3 QUIET)
|
||||||
if (${Z3_FOUND})
|
if (${Z3_FOUND})
|
||||||
@ -26,7 +26,7 @@ if (NOT (${Z3_FOUND} OR ${CVC4_FOUND}))
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_library(solidity ${sources} ${headers})
|
add_library(solidity ${sources} ${headers})
|
||||||
target_link_libraries(solidity PUBLIC evmasm langutil devcore ${Boost_FILESYSTEM_LIBRARY} ${Boost_SYSTEM_LIBRARY})
|
target_link_libraries(solidity PUBLIC yul evmasm langutil devcore ${Boost_FILESYSTEM_LIBRARY} ${Boost_SYSTEM_LIBRARY})
|
||||||
|
|
||||||
if (${Z3_FOUND})
|
if (${Z3_FOUND})
|
||||||
target_link_libraries(solidity PUBLIC ${Z3_LIBRARY})
|
target_link_libraries(solidity PUBLIC ${Z3_LIBRARY})
|
||||||
|
@ -23,12 +23,12 @@
|
|||||||
#include <libsolidity/analysis/ReferencesResolver.h>
|
#include <libsolidity/analysis/ReferencesResolver.h>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <libsolidity/analysis/NameAndTypeResolver.h>
|
#include <libsolidity/analysis/NameAndTypeResolver.h>
|
||||||
#include <liblangutil/Exceptions.h>
|
|
||||||
#include <libsolidity/analysis/ConstantEvaluator.h>
|
#include <libsolidity/analysis/ConstantEvaluator.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
#include <liblangutil/Exceptions.h>
|
||||||
|
|
||||||
#include <libdevcore/StringUtils.h>
|
#include <libdevcore/StringUtils.h>
|
||||||
|
|
||||||
@ -273,7 +273,7 @@ bool ReferencesResolver::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
ErrorList errors;
|
ErrorList errors;
|
||||||
ErrorReporter errorsIgnored(errors);
|
ErrorReporter errorsIgnored(errors);
|
||||||
yul::ExternalIdentifierAccess::Resolver resolver =
|
yul::ExternalIdentifierAccess::Resolver resolver =
|
||||||
[&](assembly::Identifier const& _identifier, yul::IdentifierContext, bool _crossesFunctionBoundary) {
|
[&](yul::Identifier const& _identifier, yul::IdentifierContext, bool _crossesFunctionBoundary) {
|
||||||
auto declarations = m_resolver.nameFromCurrentScope(_identifier.name.str());
|
auto declarations = m_resolver.nameFromCurrentScope(_identifier.name.str());
|
||||||
bool isSlot = boost::algorithm::ends_with(_identifier.name.str(), "_slot");
|
bool isSlot = boost::algorithm::ends_with(_identifier.name.str(), "_slot");
|
||||||
bool isOffset = boost::algorithm::ends_with(_identifier.name.str(), "_offset");
|
bool isOffset = boost::algorithm::ends_with(_identifier.name.str(), "_offset");
|
||||||
@ -314,9 +314,9 @@ bool ReferencesResolver::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
|
|
||||||
// Will be re-generated later with correct information
|
// Will be re-generated later with correct information
|
||||||
// We use the latest EVM version because we will re-run it anyway.
|
// We use the latest EVM version because we will re-run it anyway.
|
||||||
assembly::AsmAnalysisInfo analysisInfo;
|
yul::AsmAnalysisInfo analysisInfo;
|
||||||
boost::optional<Error::Type> errorTypeForLoose = Error::Type::SyntaxError;
|
boost::optional<Error::Type> errorTypeForLoose = Error::Type::SyntaxError;
|
||||||
assembly::AsmAnalyzer(analysisInfo, errorsIgnored, EVMVersion(), errorTypeForLoose, assembly::AsmFlavour::Loose, resolver).analyze(_inlineAssembly.operations());
|
yul::AsmAnalyzer(analysisInfo, errorsIgnored, EVMVersion(), errorTypeForLoose, yul::AsmFlavour::Loose, resolver).analyze(_inlineAssembly.operations());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,9 +27,9 @@
|
|||||||
#include <boost/algorithm/string/join.hpp>
|
#include <boost/algorithm/string/join.hpp>
|
||||||
#include <boost/range/adaptor/reversed.hpp>
|
#include <boost/range/adaptor/reversed.hpp>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
#include <libdevcore/Algorithms.h>
|
#include <libdevcore/Algorithms.h>
|
||||||
|
|
||||||
@ -954,7 +954,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
// External references have already been resolved in a prior stage and stored in the annotation.
|
// External references have already been resolved in a prior stage and stored in the annotation.
|
||||||
// We run the resolve step again regardless.
|
// We run the resolve step again regardless.
|
||||||
yul::ExternalIdentifierAccess::Resolver identifierAccess = [&](
|
yul::ExternalIdentifierAccess::Resolver identifierAccess = [&](
|
||||||
assembly::Identifier const& _identifier,
|
yul::Identifier const& _identifier,
|
||||||
yul::IdentifierContext _context,
|
yul::IdentifierContext _context,
|
||||||
bool
|
bool
|
||||||
)
|
)
|
||||||
@ -1039,13 +1039,13 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
return size_t(1);
|
return size_t(1);
|
||||||
};
|
};
|
||||||
solAssert(!_inlineAssembly.annotation().analysisInfo, "");
|
solAssert(!_inlineAssembly.annotation().analysisInfo, "");
|
||||||
_inlineAssembly.annotation().analysisInfo = make_shared<assembly::AsmAnalysisInfo>();
|
_inlineAssembly.annotation().analysisInfo = make_shared<yul::AsmAnalysisInfo>();
|
||||||
assembly::AsmAnalyzer analyzer(
|
yul::AsmAnalyzer analyzer(
|
||||||
*_inlineAssembly.annotation().analysisInfo,
|
*_inlineAssembly.annotation().analysisInfo,
|
||||||
m_errorReporter,
|
m_errorReporter,
|
||||||
m_evmVersion,
|
m_evmVersion,
|
||||||
Error::Type::SyntaxError,
|
Error::Type::SyntaxError,
|
||||||
assembly::AsmFlavour::Loose,
|
yul::AsmFlavour::Loose,
|
||||||
identifierAccess
|
identifierAccess
|
||||||
);
|
);
|
||||||
if (!analyzer.analyze(_inlineAssembly.operations()))
|
if (!analyzer.analyze(_inlineAssembly.operations()))
|
||||||
|
@ -19,8 +19,8 @@
|
|||||||
|
|
||||||
#include <libevmasm/SemanticInformation.h>
|
#include <libevmasm/SemanticInformation.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
#include <libsolidity/ast/ExperimentalFeatures.h>
|
#include <libsolidity/ast/ExperimentalFeatures.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
|
||||||
@ -40,48 +40,48 @@ public:
|
|||||||
explicit AssemblyViewPureChecker(std::function<void(StateMutability, SourceLocation const&)> _reportMutability):
|
explicit AssemblyViewPureChecker(std::function<void(StateMutability, SourceLocation const&)> _reportMutability):
|
||||||
m_reportMutability(_reportMutability) {}
|
m_reportMutability(_reportMutability) {}
|
||||||
|
|
||||||
void operator()(assembly::Label const&) { }
|
void operator()(yul::Label const&) { }
|
||||||
void operator()(assembly::Instruction const& _instruction)
|
void operator()(yul::Instruction const& _instruction)
|
||||||
{
|
{
|
||||||
checkInstruction(_instruction.location, _instruction.instruction);
|
checkInstruction(_instruction.location, _instruction.instruction);
|
||||||
}
|
}
|
||||||
void operator()(assembly::Literal const&) {}
|
void operator()(yul::Literal const&) {}
|
||||||
void operator()(assembly::Identifier const&) {}
|
void operator()(yul::Identifier const&) {}
|
||||||
void operator()(assembly::FunctionalInstruction const& _instr)
|
void operator()(yul::FunctionalInstruction const& _instr)
|
||||||
{
|
{
|
||||||
checkInstruction(_instr.location, _instr.instruction);
|
checkInstruction(_instr.location, _instr.instruction);
|
||||||
for (auto const& arg: _instr.arguments)
|
for (auto const& arg: _instr.arguments)
|
||||||
boost::apply_visitor(*this, arg);
|
boost::apply_visitor(*this, arg);
|
||||||
}
|
}
|
||||||
void operator()(assembly::ExpressionStatement const& _expr)
|
void operator()(yul::ExpressionStatement const& _expr)
|
||||||
{
|
{
|
||||||
boost::apply_visitor(*this, _expr.expression);
|
boost::apply_visitor(*this, _expr.expression);
|
||||||
}
|
}
|
||||||
void operator()(assembly::StackAssignment const&) {}
|
void operator()(yul::StackAssignment const&) {}
|
||||||
void operator()(assembly::Assignment const& _assignment)
|
void operator()(yul::Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
boost::apply_visitor(*this, *_assignment.value);
|
boost::apply_visitor(*this, *_assignment.value);
|
||||||
}
|
}
|
||||||
void operator()(assembly::VariableDeclaration const& _varDecl)
|
void operator()(yul::VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
if (_varDecl.value)
|
if (_varDecl.value)
|
||||||
boost::apply_visitor(*this, *_varDecl.value);
|
boost::apply_visitor(*this, *_varDecl.value);
|
||||||
}
|
}
|
||||||
void operator()(assembly::FunctionDefinition const& _funDef)
|
void operator()(yul::FunctionDefinition const& _funDef)
|
||||||
{
|
{
|
||||||
(*this)(_funDef.body);
|
(*this)(_funDef.body);
|
||||||
}
|
}
|
||||||
void operator()(assembly::FunctionCall const& _funCall)
|
void operator()(yul::FunctionCall const& _funCall)
|
||||||
{
|
{
|
||||||
for (auto const& arg: _funCall.arguments)
|
for (auto const& arg: _funCall.arguments)
|
||||||
boost::apply_visitor(*this, arg);
|
boost::apply_visitor(*this, arg);
|
||||||
}
|
}
|
||||||
void operator()(assembly::If const& _if)
|
void operator()(yul::If const& _if)
|
||||||
{
|
{
|
||||||
boost::apply_visitor(*this, *_if.condition);
|
boost::apply_visitor(*this, *_if.condition);
|
||||||
(*this)(_if.body);
|
(*this)(_if.body);
|
||||||
}
|
}
|
||||||
void operator()(assembly::Switch const& _switch)
|
void operator()(yul::Switch const& _switch)
|
||||||
{
|
{
|
||||||
boost::apply_visitor(*this, *_switch.expression);
|
boost::apply_visitor(*this, *_switch.expression);
|
||||||
for (auto const& _case: _switch.cases)
|
for (auto const& _case: _switch.cases)
|
||||||
@ -91,14 +91,14 @@ public:
|
|||||||
(*this)(_case.body);
|
(*this)(_case.body);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void operator()(assembly::ForLoop const& _for)
|
void operator()(yul::ForLoop const& _for)
|
||||||
{
|
{
|
||||||
(*this)(_for.pre);
|
(*this)(_for.pre);
|
||||||
boost::apply_visitor(*this, *_for.condition);
|
boost::apply_visitor(*this, *_for.condition);
|
||||||
(*this)(_for.body);
|
(*this)(_for.body);
|
||||||
(*this)(_for.post);
|
(*this)(_for.post);
|
||||||
}
|
}
|
||||||
void operator()(assembly::Block const& _block)
|
void operator()(yul::Block const& _block)
|
||||||
{
|
{
|
||||||
for (auto const& s: _block.statements)
|
for (auto const& s: _block.statements)
|
||||||
boost::apply_visitor(*this, s);
|
boost::apply_visitor(*this, s);
|
||||||
|
@ -41,6 +41,12 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
|
namespace yul
|
||||||
|
{
|
||||||
|
// Forward-declaration to <yul/AsmData.h>
|
||||||
|
struct Block;
|
||||||
|
}
|
||||||
|
|
||||||
namespace dev
|
namespace dev
|
||||||
{
|
{
|
||||||
namespace solidity
|
namespace solidity
|
||||||
@ -1028,12 +1034,6 @@ public:
|
|||||||
StatementAnnotation& annotation() const override;
|
StatementAnnotation& annotation() const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace assembly
|
|
||||||
{
|
|
||||||
// Forward-declaration to AsmData.h
|
|
||||||
struct Block;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Inline assembly.
|
* Inline assembly.
|
||||||
*/
|
*/
|
||||||
@ -1043,18 +1043,18 @@ public:
|
|||||||
InlineAssembly(
|
InlineAssembly(
|
||||||
SourceLocation const& _location,
|
SourceLocation const& _location,
|
||||||
ASTPointer<ASTString> const& _docString,
|
ASTPointer<ASTString> const& _docString,
|
||||||
std::shared_ptr<assembly::Block> const& _operations
|
std::shared_ptr<yul::Block> const& _operations
|
||||||
):
|
):
|
||||||
Statement(_location, _docString), m_operations(_operations) {}
|
Statement(_location, _docString), m_operations(_operations) {}
|
||||||
void accept(ASTVisitor& _visitor) override;
|
void accept(ASTVisitor& _visitor) override;
|
||||||
void accept(ASTConstVisitor& _visitor) const override;
|
void accept(ASTConstVisitor& _visitor) const override;
|
||||||
|
|
||||||
assembly::Block const& operations() const { return *m_operations; }
|
yul::Block const& operations() const { return *m_operations; }
|
||||||
|
|
||||||
InlineAssemblyAnnotation& annotation() const override;
|
InlineAssemblyAnnotation& annotation() const override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::shared_ptr<assembly::Block> m_operations;
|
std::shared_ptr<yul::Block> m_operations;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -30,6 +30,12 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
|
namespace yul
|
||||||
|
{
|
||||||
|
struct AsmAnalysisInfo;
|
||||||
|
struct Identifier;
|
||||||
|
}
|
||||||
|
|
||||||
namespace dev
|
namespace dev
|
||||||
{
|
{
|
||||||
namespace solidity
|
namespace solidity
|
||||||
@ -120,12 +126,6 @@ struct StatementAnnotation: ASTAnnotation, DocumentedAnnotation
|
|||||||
{
|
{
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace assembly
|
|
||||||
{
|
|
||||||
struct AsmAnalysisInfo;
|
|
||||||
struct Identifier;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct InlineAssemblyAnnotation: StatementAnnotation
|
struct InlineAssemblyAnnotation: StatementAnnotation
|
||||||
{
|
{
|
||||||
struct ExternalIdentifierInfo
|
struct ExternalIdentifierInfo
|
||||||
@ -137,9 +137,9 @@ struct InlineAssemblyAnnotation: StatementAnnotation
|
|||||||
};
|
};
|
||||||
|
|
||||||
/// Mapping containing resolved references to external identifiers and their value size
|
/// Mapping containing resolved references to external identifiers and their value size
|
||||||
std::map<assembly::Identifier const*, ExternalIdentifierInfo> externalReferences;
|
std::map<yul::Identifier const*, ExternalIdentifierInfo> externalReferences;
|
||||||
/// Information generated during analysis phase.
|
/// Information generated during analysis phase.
|
||||||
std::shared_ptr<assembly::AsmAnalysisInfo> analysisInfo;
|
std::shared_ptr<yul::AsmAnalysisInfo> analysisInfo;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ReturnAnnotation: StatementAnnotation
|
struct ReturnAnnotation: StatementAnnotation
|
||||||
|
@ -23,8 +23,8 @@
|
|||||||
#include <boost/algorithm/string/join.hpp>
|
#include <boost/algorithm/string/join.hpp>
|
||||||
#include <libdevcore/UTF8.h>
|
#include <libdevcore/UTF8.h>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
@ -172,7 +172,7 @@ void ASTJsonConverter::appendExpressionAttributes(
|
|||||||
_attributes += exprAttributes;
|
_attributes += exprAttributes;
|
||||||
}
|
}
|
||||||
|
|
||||||
Json::Value ASTJsonConverter::inlineAssemblyIdentifierToJson(pair<assembly::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->location);
|
||||||
@ -465,7 +465,7 @@ bool ASTJsonConverter::visit(InlineAssembly const& _node)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
setJsonNode(_node, "InlineAssembly", {
|
setJsonNode(_node, "InlineAssembly", {
|
||||||
make_pair("operations", Json::Value(assembly::AsmPrinter()(_node.operations()))),
|
make_pair("operations", Json::Value(yul::AsmPrinter()(_node.operations()))),
|
||||||
make_pair("externalReferences", std::move(externalReferences))
|
make_pair("externalReferences", std::move(externalReferences))
|
||||||
});
|
});
|
||||||
return false;
|
return false;
|
||||||
|
@ -135,7 +135,7 @@ private:
|
|||||||
{
|
{
|
||||||
return _node ? toJson(*_node) : Json::nullValue;
|
return _node ? toJson(*_node) : Json::nullValue;
|
||||||
}
|
}
|
||||||
Json::Value inlineAssemblyIdentifierToJson(std::pair<assembly::Identifier const* , InlineAssemblyAnnotation::ExternalIdentifierInfo> _info) const;
|
Json::Value inlineAssemblyIdentifierToJson(std::pair<yul::Identifier const* , InlineAssemblyAnnotation::ExternalIdentifierInfo> _info) const;
|
||||||
static std::string location(VariableDeclaration::Location _location);
|
static std::string location(VariableDeclaration::Location _location);
|
||||||
static std::string contractKind(ContractDefinition::ContractKind _kind);
|
static std::string contractKind(ContractDefinition::ContractKind _kind);
|
||||||
static std::string functionCallKind(FunctionCallKind _kind);
|
static std::string functionCallKind(FunctionCallKind _kind);
|
||||||
|
@ -25,15 +25,14 @@
|
|||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <libsolidity/codegen/Compiler.h>
|
#include <libsolidity/codegen/Compiler.h>
|
||||||
#include <libsolidity/interface/Version.h>
|
#include <libsolidity/interface/Version.h>
|
||||||
|
#include <libsolidity/interface/SourceReferenceFormatter.h>
|
||||||
|
#include <libyul/AsmParser.h>
|
||||||
|
#include <libyul/AsmCodeGen.h>
|
||||||
|
#include <libyul/AsmAnalysis.h>
|
||||||
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
#include <libyul/YulString.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <libsolidity/interface/SourceReferenceFormatter.h>
|
|
||||||
#include <liblangutil/Scanner.h>
|
|
||||||
#include <libsolidity/inlineasm/AsmParser.h>
|
|
||||||
#include <libsolidity/inlineasm/AsmCodeGen.h>
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
|
||||||
#include <libyul/YulString.h>
|
|
||||||
|
|
||||||
#include <boost/algorithm/string/replace.hpp>
|
#include <boost/algorithm/string/replace.hpp>
|
||||||
|
|
||||||
@ -43,7 +42,7 @@
|
|||||||
// Change to "define" to output all intermediate code
|
// Change to "define" to output all intermediate code
|
||||||
#undef SOL_OUTPUT_ASM
|
#undef SOL_OUTPUT_ASM
|
||||||
#ifdef SOL_OUTPUT_ASM
|
#ifdef SOL_OUTPUT_ASM
|
||||||
#include <libsolidity/inlineasm/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
@ -324,7 +323,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
|
|
||||||
yul::ExternalIdentifierAccess identifierAccess;
|
yul::ExternalIdentifierAccess identifierAccess;
|
||||||
identifierAccess.resolve = [&](
|
identifierAccess.resolve = [&](
|
||||||
assembly::Identifier const& _identifier,
|
yul::Identifier const& _identifier,
|
||||||
yul::IdentifierContext,
|
yul::IdentifierContext,
|
||||||
bool
|
bool
|
||||||
)
|
)
|
||||||
@ -333,7 +332,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
return it == _localVariables.end() ? size_t(-1) : 1;
|
return it == _localVariables.end() ? size_t(-1) : 1;
|
||||||
};
|
};
|
||||||
identifierAccess.generateCode = [&](
|
identifierAccess.generateCode = [&](
|
||||||
assembly::Identifier const& _identifier,
|
yul::Identifier const& _identifier,
|
||||||
yul::IdentifierContext _context,
|
yul::IdentifierContext _context,
|
||||||
yul::AbstractAssembly& _assembly
|
yul::AbstractAssembly& _assembly
|
||||||
)
|
)
|
||||||
@ -362,19 +361,19 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
ErrorList errors;
|
ErrorList errors;
|
||||||
ErrorReporter errorReporter(errors);
|
ErrorReporter errorReporter(errors);
|
||||||
auto scanner = make_shared<langutil::Scanner>(langutil::CharStream(_assembly), "--CODEGEN--");
|
auto scanner = make_shared<langutil::Scanner>(langutil::CharStream(_assembly), "--CODEGEN--");
|
||||||
auto parserResult = assembly::Parser(errorReporter, assembly::AsmFlavour::Strict).parse(scanner, false);
|
auto parserResult = yul::Parser(errorReporter, yul::AsmFlavour::Strict).parse(scanner, false);
|
||||||
#ifdef SOL_OUTPUT_ASM
|
#ifdef SOL_OUTPUT_ASM
|
||||||
cout << assembly::AsmPrinter()(*parserResult) << endl;
|
cout << yul::AsmPrinter()(*parserResult) << endl;
|
||||||
#endif
|
#endif
|
||||||
assembly::AsmAnalysisInfo analysisInfo;
|
yul::AsmAnalysisInfo analysisInfo;
|
||||||
bool analyzerResult = false;
|
bool analyzerResult = false;
|
||||||
if (parserResult)
|
if (parserResult)
|
||||||
analyzerResult = assembly::AsmAnalyzer(
|
analyzerResult = yul::AsmAnalyzer(
|
||||||
analysisInfo,
|
analysisInfo,
|
||||||
errorReporter,
|
errorReporter,
|
||||||
m_evmVersion,
|
m_evmVersion,
|
||||||
boost::none,
|
boost::none,
|
||||||
assembly::AsmFlavour::Strict,
|
yul::AsmFlavour::Strict,
|
||||||
identifierAccess.resolve
|
identifierAccess.resolve
|
||||||
).analyze(*parserResult);
|
).analyze(*parserResult);
|
||||||
if (!parserResult || !errorReporter.errors().empty() || !analyzerResult)
|
if (!parserResult || !errorReporter.errors().empty() || !analyzerResult)
|
||||||
@ -396,7 +395,7 @@ void CompilerContext::appendInlineAssembly(
|
|||||||
}
|
}
|
||||||
|
|
||||||
solAssert(errorReporter.errors().empty(), "Failed to analyze inline assembly block.");
|
solAssert(errorReporter.errors().empty(), "Failed to analyze inline assembly block.");
|
||||||
assembly::CodeGenerator::assemble(*parserResult, analysisInfo, *m_asm, identifierAccess, _system);
|
yul::CodeGenerator::assemble(*parserResult, analysisInfo, *m_asm, identifierAccess, _system);
|
||||||
|
|
||||||
// Reset the source location to the one of the node (instead of the CODEGEN source location)
|
// Reset the source location to the one of the node (instead of the CODEGEN source location)
|
||||||
updateSourceLocation();
|
updateSourceLocation();
|
||||||
|
@ -21,11 +21,11 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/codegen/ContractCompiler.h>
|
#include <libsolidity/codegen/ContractCompiler.h>
|
||||||
#include <libsolidity/inlineasm/AsmCodeGen.h>
|
|
||||||
#include <libsolidity/ast/AST.h>
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
|
||||||
#include <libsolidity/codegen/ExpressionCompiler.h>
|
#include <libsolidity/codegen/ExpressionCompiler.h>
|
||||||
#include <libsolidity/codegen/CompilerUtils.h>
|
#include <libsolidity/codegen/CompilerUtils.h>
|
||||||
|
#include <libsolidity/ast/AST.h>
|
||||||
|
#include <libyul/AsmCodeGen.h>
|
||||||
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
|
||||||
#include <libevmasm/Instruction.h>
|
#include <libevmasm/Instruction.h>
|
||||||
#include <libevmasm/Assembly.h>
|
#include <libevmasm/Assembly.h>
|
||||||
@ -496,14 +496,14 @@ bool ContractCompiler::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
unsigned startStackHeight = m_context.stackHeight();
|
unsigned startStackHeight = m_context.stackHeight();
|
||||||
yul::ExternalIdentifierAccess identifierAccess;
|
yul::ExternalIdentifierAccess identifierAccess;
|
||||||
identifierAccess.resolve = [&](assembly::Identifier const& _identifier, yul::IdentifierContext, bool)
|
identifierAccess.resolve = [&](yul::Identifier const& _identifier, yul::IdentifierContext, bool)
|
||||||
{
|
{
|
||||||
auto ref = _inlineAssembly.annotation().externalReferences.find(&_identifier);
|
auto ref = _inlineAssembly.annotation().externalReferences.find(&_identifier);
|
||||||
if (ref == _inlineAssembly.annotation().externalReferences.end())
|
if (ref == _inlineAssembly.annotation().externalReferences.end())
|
||||||
return size_t(-1);
|
return size_t(-1);
|
||||||
return ref->second.valueSize;
|
return ref->second.valueSize;
|
||||||
};
|
};
|
||||||
identifierAccess.generateCode = [&](assembly::Identifier const& _identifier, yul::IdentifierContext _context, yul::AbstractAssembly& _assembly)
|
identifierAccess.generateCode = [&](yul::Identifier const& _identifier, yul::IdentifierContext _context, yul::AbstractAssembly& _assembly)
|
||||||
{
|
{
|
||||||
auto ref = _inlineAssembly.annotation().externalReferences.find(&_identifier);
|
auto ref = _inlineAssembly.annotation().externalReferences.find(&_identifier);
|
||||||
solAssert(ref != _inlineAssembly.annotation().externalReferences.end(), "");
|
solAssert(ref != _inlineAssembly.annotation().externalReferences.end(), "");
|
||||||
@ -615,7 +615,7 @@ bool ContractCompiler::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
solAssert(_inlineAssembly.annotation().analysisInfo, "");
|
solAssert(_inlineAssembly.annotation().analysisInfo, "");
|
||||||
assembly::CodeGenerator::assemble(
|
yul::CodeGenerator::assemble(
|
||||||
_inlineAssembly.operations(),
|
_inlineAssembly.operations(),
|
||||||
*_inlineAssembly.annotation().analysisInfo,
|
*_inlineAssembly.annotation().analysisInfo,
|
||||||
m_context.nonConstAssembly(),
|
m_context.nonConstAssembly(),
|
||||||
|
@ -1,68 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of solidity.
|
|
||||||
|
|
||||||
solidity is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
solidity is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
/**
|
|
||||||
* @author Christian <c@ethdev.com>
|
|
||||||
* @date 2017
|
|
||||||
* Converts a parsed assembly into its textual form.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
|
||||||
|
|
||||||
class AsmPrinter: public boost::static_visitor<std::string>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit AsmPrinter(bool _yul = false): m_yul(_yul) {}
|
|
||||||
|
|
||||||
std::string operator()(assembly::Instruction const& _instruction);
|
|
||||||
std::string operator()(assembly::Literal const& _literal);
|
|
||||||
std::string operator()(assembly::Identifier const& _identifier);
|
|
||||||
std::string operator()(assembly::FunctionalInstruction const& _functionalInstruction);
|
|
||||||
std::string operator()(assembly::ExpressionStatement const& _expr);
|
|
||||||
std::string operator()(assembly::Label const& _label);
|
|
||||||
std::string operator()(assembly::StackAssignment const& _assignment);
|
|
||||||
std::string operator()(assembly::Assignment const& _assignment);
|
|
||||||
std::string operator()(assembly::VariableDeclaration const& _variableDeclaration);
|
|
||||||
std::string operator()(assembly::FunctionDefinition const& _functionDefinition);
|
|
||||||
std::string operator()(assembly::FunctionCall const& _functionCall);
|
|
||||||
std::string operator()(assembly::If const& _if);
|
|
||||||
std::string operator()(assembly::Switch const& _switch);
|
|
||||||
std::string operator()(assembly::ForLoop const& _forLoop);
|
|
||||||
std::string operator()(assembly::Block const& _block);
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::string formatTypedName(TypedName _variable) const;
|
|
||||||
std::string appendTypeName(yul::YulString _type) const;
|
|
||||||
|
|
||||||
bool m_yul = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -23,17 +23,16 @@
|
|||||||
#include <libsolidity/interface/AssemblyStack.h>
|
#include <libsolidity/interface/AssemblyStack.h>
|
||||||
|
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <libsolidity/inlineasm/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libsolidity/inlineasm/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libsolidity/inlineasm/AsmCodeGen.h>
|
#include <libyul/AsmCodeGen.h>
|
||||||
|
|
||||||
#include <libevmasm/Assembly.h>
|
|
||||||
|
|
||||||
#include <libyul/backends/evm/EVMCodeTransform.h>
|
#include <libyul/backends/evm/EVMCodeTransform.h>
|
||||||
#include <libyul/backends/evm/EVMAssembly.h>
|
#include <libyul/backends/evm/EVMAssembly.h>
|
||||||
|
|
||||||
|
#include <libevmasm/Assembly.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
@ -41,19 +40,19 @@ using namespace dev::solidity;
|
|||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
assembly::AsmFlavour languageToAsmFlavour(AssemblyStack::Language _language)
|
yul::AsmFlavour languageToAsmFlavour(AssemblyStack::Language _language)
|
||||||
{
|
{
|
||||||
switch (_language)
|
switch (_language)
|
||||||
{
|
{
|
||||||
case AssemblyStack::Language::Assembly:
|
case AssemblyStack::Language::Assembly:
|
||||||
return assembly::AsmFlavour::Loose;
|
return yul::AsmFlavour::Loose;
|
||||||
case AssemblyStack::Language::StrictAssembly:
|
case AssemblyStack::Language::StrictAssembly:
|
||||||
return assembly::AsmFlavour::Strict;
|
return yul::AsmFlavour::Strict;
|
||||||
case AssemblyStack::Language::Yul:
|
case AssemblyStack::Language::Yul:
|
||||||
return assembly::AsmFlavour::Yul;
|
return yul::AsmFlavour::Yul;
|
||||||
}
|
}
|
||||||
solAssert(false, "");
|
solAssert(false, "");
|
||||||
return assembly::AsmFlavour::Yul;
|
return yul::AsmFlavour::Yul;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -70,7 +69,7 @@ bool AssemblyStack::parseAndAnalyze(std::string const& _sourceName, std::string
|
|||||||
m_errors.clear();
|
m_errors.clear();
|
||||||
m_analysisSuccessful = false;
|
m_analysisSuccessful = false;
|
||||||
m_scanner = make_shared<Scanner>(CharStream(_source), _sourceName);
|
m_scanner = make_shared<Scanner>(CharStream(_source), _sourceName);
|
||||||
m_parserResult = assembly::Parser(m_errorReporter, languageToAsmFlavour(m_language)).parse(m_scanner, false);
|
m_parserResult = yul::Parser(m_errorReporter, languageToAsmFlavour(m_language)).parse(m_scanner, false);
|
||||||
if (!m_errorReporter.errors().empty())
|
if (!m_errorReporter.errors().empty())
|
||||||
return false;
|
return false;
|
||||||
solAssert(m_parserResult, "");
|
solAssert(m_parserResult, "");
|
||||||
@ -78,21 +77,21 @@ bool AssemblyStack::parseAndAnalyze(std::string const& _sourceName, std::string
|
|||||||
return analyzeParsed();
|
return analyzeParsed();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AssemblyStack::analyze(assembly::Block const& _block, Scanner const* _scanner)
|
bool AssemblyStack::analyze(yul::Block const& _block, Scanner const* _scanner)
|
||||||
{
|
{
|
||||||
m_errors.clear();
|
m_errors.clear();
|
||||||
m_analysisSuccessful = false;
|
m_analysisSuccessful = false;
|
||||||
if (_scanner)
|
if (_scanner)
|
||||||
m_scanner = make_shared<Scanner>(*_scanner);
|
m_scanner = make_shared<Scanner>(*_scanner);
|
||||||
m_parserResult = make_shared<assembly::Block>(_block);
|
m_parserResult = make_shared<yul::Block>(_block);
|
||||||
|
|
||||||
return analyzeParsed();
|
return analyzeParsed();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AssemblyStack::analyzeParsed()
|
bool AssemblyStack::analyzeParsed()
|
||||||
{
|
{
|
||||||
m_analysisInfo = make_shared<assembly::AsmAnalysisInfo>();
|
m_analysisInfo = make_shared<yul::AsmAnalysisInfo>();
|
||||||
assembly::AsmAnalyzer analyzer(*m_analysisInfo, m_errorReporter, m_evmVersion, boost::none, languageToAsmFlavour(m_language));
|
yul::AsmAnalyzer analyzer(*m_analysisInfo, m_errorReporter, m_evmVersion, boost::none, languageToAsmFlavour(m_language));
|
||||||
m_analysisSuccessful = analyzer.analyze(*m_parserResult);
|
m_analysisSuccessful = analyzer.analyze(*m_parserResult);
|
||||||
return m_analysisSuccessful;
|
return m_analysisSuccessful;
|
||||||
}
|
}
|
||||||
@ -109,7 +108,7 @@ MachineAssemblyObject AssemblyStack::assemble(Machine _machine) const
|
|||||||
{
|
{
|
||||||
MachineAssemblyObject object;
|
MachineAssemblyObject object;
|
||||||
eth::Assembly assembly;
|
eth::Assembly assembly;
|
||||||
assembly::CodeGenerator::assemble(*m_parserResult, *m_analysisInfo, assembly);
|
yul::CodeGenerator::assemble(*m_parserResult, *m_analysisInfo, assembly);
|
||||||
object.bytecode = make_shared<eth::LinkerObject>(assembly.assemble());
|
object.bytecode = make_shared<eth::LinkerObject>(assembly.assemble());
|
||||||
object.assembly = assembly.assemblyString();
|
object.assembly = assembly.assemblyString();
|
||||||
return object;
|
return object;
|
||||||
@ -133,5 +132,5 @@ MachineAssemblyObject AssemblyStack::assemble(Machine _machine) const
|
|||||||
string AssemblyStack::print() const
|
string AssemblyStack::print() const
|
||||||
{
|
{
|
||||||
solAssert(m_parserResult, "");
|
solAssert(m_parserResult, "");
|
||||||
return assembly::AsmPrinter(m_language == Language::Yul)(*m_parserResult);
|
return yul::AsmPrinter(m_language == Language::Yul)(*m_parserResult);
|
||||||
}
|
}
|
||||||
|
@ -34,16 +34,17 @@ namespace langutil
|
|||||||
class Scanner;
|
class Scanner;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
struct AsmAnalysisInfo;
|
struct AsmAnalysisInfo;
|
||||||
struct Block;
|
struct Block;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace dev
|
||||||
|
{
|
||||||
|
namespace solidity
|
||||||
|
{
|
||||||
|
|
||||||
struct MachineAssemblyObject
|
struct MachineAssemblyObject
|
||||||
{
|
{
|
||||||
std::shared_ptr<eth::LinkerObject> bytecode;
|
std::shared_ptr<eth::LinkerObject> bytecode;
|
||||||
@ -73,7 +74,7 @@ public:
|
|||||||
|
|
||||||
/// Runs analysis step on the supplied block, returns false if input cannot be assembled.
|
/// Runs analysis step on the supplied block, returns false if input cannot be assembled.
|
||||||
/// Multiple calls overwrite the previous state.
|
/// Multiple calls overwrite the previous state.
|
||||||
bool analyze(assembly::Block const& _block, langutil::Scanner const* _scanner = nullptr);
|
bool analyze(yul::Block const& _block, langutil::Scanner const* _scanner = nullptr);
|
||||||
|
|
||||||
/// Run the assembly step (should only be called after parseAndAnalyze).
|
/// Run the assembly step (should only be called after parseAndAnalyze).
|
||||||
MachineAssemblyObject assemble(Machine _machine) const;
|
MachineAssemblyObject assemble(Machine _machine) const;
|
||||||
@ -93,8 +94,8 @@ private:
|
|||||||
std::shared_ptr<langutil::Scanner> m_scanner;
|
std::shared_ptr<langutil::Scanner> m_scanner;
|
||||||
|
|
||||||
bool m_analysisSuccessful = false;
|
bool m_analysisSuccessful = false;
|
||||||
std::shared_ptr<assembly::Block> m_parserResult;
|
std::shared_ptr<yul::Block> m_parserResult;
|
||||||
std::shared_ptr<assembly::AsmAnalysisInfo> m_analysisInfo;
|
std::shared_ptr<yul::AsmAnalysisInfo> m_analysisInfo;
|
||||||
langutil::ErrorList m_errors;
|
langutil::ErrorList m_errors;
|
||||||
langutil::ErrorReporter m_errorReporter;
|
langutil::ErrorReporter m_errorReporter;
|
||||||
};
|
};
|
||||||
|
@ -27,7 +27,6 @@
|
|||||||
#include <libsolidity/interface/Version.h>
|
#include <libsolidity/interface/Version.h>
|
||||||
#include <libsolidity/analysis/SemVerHandler.h>
|
#include <libsolidity/analysis/SemVerHandler.h>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
#include <liblangutil/Scanner.h>
|
|
||||||
#include <libsolidity/parsing/Parser.h>
|
#include <libsolidity/parsing/Parser.h>
|
||||||
#include <libsolidity/analysis/ControlFlowAnalyzer.h>
|
#include <libsolidity/analysis/ControlFlowAnalyzer.h>
|
||||||
#include <libsolidity/analysis/ControlFlowGraph.h>
|
#include <libsolidity/analysis/ControlFlowGraph.h>
|
||||||
@ -45,10 +44,12 @@
|
|||||||
#include <libsolidity/interface/Natspec.h>
|
#include <libsolidity/interface/Natspec.h>
|
||||||
#include <libsolidity/interface/GasEstimator.h>
|
#include <libsolidity/interface/GasEstimator.h>
|
||||||
|
|
||||||
#include <libevmasm/Exceptions.h>
|
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
|
#include <liblangutil/Scanner.h>
|
||||||
|
|
||||||
|
#include <libevmasm/Exceptions.h>
|
||||||
|
|
||||||
#include <libdevcore/SwarmHash.h>
|
#include <libdevcore/SwarmHash.h>
|
||||||
#include <libdevcore/JSON.h>
|
#include <libdevcore/JSON.h>
|
||||||
|
|
||||||
|
@ -22,11 +22,11 @@
|
|||||||
|
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <liblangutil/SourceLocation.h>
|
|
||||||
#include <libsolidity/parsing/Parser.h>
|
#include <libsolidity/parsing/Parser.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libsolidity/inlineasm/AsmParser.h>
|
#include <liblangutil/SourceLocation.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
#include <liblangutil/Scanner.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
@ -1012,8 +1012,8 @@ ASTPointer<InlineAssembly> Parser::parseInlineAssembly(ASTPointer<ASTString> con
|
|||||||
m_scanner->next();
|
m_scanner->next();
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Parser asmParser(m_errorReporter);
|
yul::Parser asmParser(m_errorReporter);
|
||||||
shared_ptr<assembly::Block> block = asmParser.parse(m_scanner, true);
|
shared_ptr<yul::Block> block = asmParser.parse(m_scanner, true);
|
||||||
nodeFactory.markEndPosition();
|
nodeFactory.markEndPosition();
|
||||||
return nodeFactory.createNode<InlineAssembly>(_docString, block);
|
return nodeFactory.createNode<InlineAssembly>(_docString, block);
|
||||||
}
|
}
|
||||||
|
@ -1,55 +0,0 @@
|
|||||||
/*
|
|
||||||
This file is part of solidity.
|
|
||||||
|
|
||||||
solidity is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
solidity is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
/**
|
|
||||||
* @date 2017
|
|
||||||
* Pull in some identifiers from the solidity::assembly namespace.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
|
||||||
{
|
|
||||||
|
|
||||||
using Instruction = solidity::assembly::Instruction;
|
|
||||||
using Literal = solidity::assembly::Literal;
|
|
||||||
using Label = solidity::assembly::Label;
|
|
||||||
using StackAssignment = solidity::assembly::StackAssignment;
|
|
||||||
using Identifier = solidity::assembly::Identifier;
|
|
||||||
using Assignment = solidity::assembly::Assignment;
|
|
||||||
using VariableDeclaration = solidity::assembly::VariableDeclaration;
|
|
||||||
using FunctionalInstruction = solidity::assembly::FunctionalInstruction;
|
|
||||||
using FunctionDefinition = solidity::assembly::FunctionDefinition;
|
|
||||||
using FunctionCall = solidity::assembly::FunctionCall;
|
|
||||||
using If = solidity::assembly::If;
|
|
||||||
using Case = solidity::assembly::Case;
|
|
||||||
using Switch = solidity::assembly::Switch;
|
|
||||||
using ForLoop = solidity::assembly::ForLoop;
|
|
||||||
using ExpressionStatement = solidity::assembly::ExpressionStatement;
|
|
||||||
using Block = solidity::assembly::Block;
|
|
||||||
|
|
||||||
using TypedName = solidity::assembly::TypedName;
|
|
||||||
class YulString;
|
|
||||||
|
|
||||||
using Expression = boost::variant<FunctionalInstruction, FunctionCall, Identifier, Literal>;
|
|
||||||
using Statement = boost::variant<ExpressionStatement, Instruction, Label, StackAssignment, Assignment, VariableDeclaration, FunctionDefinition, If, Switch, ForLoop, Block>;
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
@ -18,12 +18,12 @@
|
|||||||
* Analyzer part of inline assembly.
|
* Analyzer part of inline assembly.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmScopeFiller.h>
|
#include <libyul/AsmScopeFiller.h>
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
|
||||||
@ -36,8 +36,9 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
|
using namespace yul;
|
||||||
|
using namespace dev;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
using namespace dev::solidity::assembly;
|
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -65,7 +66,7 @@ bool AsmAnalyzer::operator()(Label const& _label)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::Instruction const& _instruction)
|
bool AsmAnalyzer::operator()(yul::Instruction const& _instruction)
|
||||||
{
|
{
|
||||||
checkLooseFeature(
|
checkLooseFeature(
|
||||||
_instruction.location,
|
_instruction.location,
|
||||||
@ -78,11 +79,11 @@ bool AsmAnalyzer::operator()(assembly::Instruction const& _instruction)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::Literal const& _literal)
|
bool AsmAnalyzer::operator()(Literal const& _literal)
|
||||||
{
|
{
|
||||||
expectValidType(_literal.type.str(), _literal.location);
|
expectValidType(_literal.type.str(), _literal.location);
|
||||||
++m_stackHeight;
|
++m_stackHeight;
|
||||||
if (_literal.kind == assembly::LiteralKind::String && _literal.value.str().size() > 32)
|
if (_literal.kind == LiteralKind::String && _literal.value.str().size() > 32)
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
_literal.location,
|
_literal.location,
|
||||||
@ -90,7 +91,7 @@ bool AsmAnalyzer::operator()(assembly::Literal const& _literal)
|
|||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (_literal.kind == assembly::LiteralKind::Number && bigint(_literal.value.str()) > u256(-1))
|
else if (_literal.kind == LiteralKind::Number && bigint(_literal.value.str()) > u256(-1))
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(
|
m_errorReporter.typeError(
|
||||||
_literal.location,
|
_literal.location,
|
||||||
@ -98,7 +99,7 @@ bool AsmAnalyzer::operator()(assembly::Literal const& _literal)
|
|||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else if (_literal.kind == assembly::LiteralKind::Boolean)
|
else if (_literal.kind == LiteralKind::Boolean)
|
||||||
{
|
{
|
||||||
solAssert(m_flavour == AsmFlavour::Yul, "");
|
solAssert(m_flavour == AsmFlavour::Yul, "");
|
||||||
solAssert(_literal.value == YulString{string("true")} || _literal.value == YulString{string("false")}, "");
|
solAssert(_literal.value == YulString{string("true")} || _literal.value == YulString{string("false")}, "");
|
||||||
@ -107,7 +108,7 @@ bool AsmAnalyzer::operator()(assembly::Literal const& _literal)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::Identifier const& _identifier)
|
bool AsmAnalyzer::operator()(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
solAssert(!_identifier.name.empty(), "");
|
solAssert(!_identifier.name.empty(), "");
|
||||||
size_t numErrorsBefore = m_errorReporter.errors().size();
|
size_t numErrorsBefore = m_errorReporter.errors().size();
|
||||||
@ -177,7 +178,7 @@ bool AsmAnalyzer::operator()(FunctionalInstruction const& _instr)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::ExpressionStatement const& _statement)
|
bool AsmAnalyzer::operator()(ExpressionStatement const& _statement)
|
||||||
{
|
{
|
||||||
int initialStackHeight = m_stackHeight;
|
int initialStackHeight = m_stackHeight;
|
||||||
bool success = boost::apply_visitor(*this, _statement.expression);
|
bool success = boost::apply_visitor(*this, _statement.expression);
|
||||||
@ -198,7 +199,7 @@ bool AsmAnalyzer::operator()(assembly::ExpressionStatement const& _statement)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::StackAssignment const& _assignment)
|
bool AsmAnalyzer::operator()(StackAssignment const& _assignment)
|
||||||
{
|
{
|
||||||
checkLooseFeature(
|
checkLooseFeature(
|
||||||
_assignment.location,
|
_assignment.location,
|
||||||
@ -209,7 +210,7 @@ bool AsmAnalyzer::operator()(assembly::StackAssignment const& _assignment)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::Assignment const& _assignment)
|
bool AsmAnalyzer::operator()(Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
solAssert(_assignment.value, "");
|
solAssert(_assignment.value, "");
|
||||||
int const expectedItems = _assignment.variableNames.size();
|
int const expectedItems = _assignment.variableNames.size();
|
||||||
@ -235,7 +236,7 @@ bool AsmAnalyzer::operator()(assembly::Assignment const& _assignment)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::VariableDeclaration const& _varDecl)
|
bool AsmAnalyzer::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
bool success = true;
|
bool success = true;
|
||||||
int const numVariables = _varDecl.variables.size();
|
int const numVariables = _varDecl.variables.size();
|
||||||
@ -261,7 +262,7 @@ bool AsmAnalyzer::operator()(assembly::VariableDeclaration const& _varDecl)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::FunctionDefinition const& _funDef)
|
bool AsmAnalyzer::operator()(FunctionDefinition const& _funDef)
|
||||||
{
|
{
|
||||||
solAssert(!_funDef.name.empty(), "");
|
solAssert(!_funDef.name.empty(), "");
|
||||||
Block const* virtualBlock = m_info.virtualBlocks.at(&_funDef).get();
|
Block const* virtualBlock = m_info.virtualBlocks.at(&_funDef).get();
|
||||||
@ -283,7 +284,7 @@ bool AsmAnalyzer::operator()(assembly::FunctionDefinition const& _funDef)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::FunctionCall const& _funCall)
|
bool AsmAnalyzer::operator()(FunctionCall const& _funCall)
|
||||||
{
|
{
|
||||||
solAssert(!_funCall.functionName.name.empty(), "");
|
solAssert(!_funCall.functionName.name.empty(), "");
|
||||||
bool success = true;
|
bool success = true;
|
||||||
@ -397,7 +398,7 @@ bool AsmAnalyzer::operator()(Switch const& _switch)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::operator()(assembly::ForLoop const& _for)
|
bool AsmAnalyzer::operator()(ForLoop const& _for)
|
||||||
{
|
{
|
||||||
solAssert(_for.condition, "");
|
solAssert(_for.condition, "");
|
||||||
|
|
||||||
@ -486,7 +487,7 @@ bool AsmAnalyzer::expectDeposit(int _deposit, int _oldHeight, SourceLocation con
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AsmAnalyzer::checkAssignment(assembly::Identifier const& _variable, size_t _valueSize)
|
bool AsmAnalyzer::checkAssignment(Identifier const& _variable, size_t _valueSize)
|
||||||
{
|
{
|
||||||
solAssert(!_variable.name.empty(), "");
|
solAssert(!_variable.name.empty(), "");
|
||||||
bool success = true;
|
bool success = true;
|
@ -23,11 +23,11 @@
|
|||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
#include <liblangutil/EVMVersion.h>
|
#include <liblangutil/EVMVersion.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
|
|
||||||
#include <libyul/backends/evm/AbstractAssembly.h>
|
#include <libyul/backends/evm/AbstractAssembly.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
@ -41,11 +41,7 @@ class ErrorReporter;
|
|||||||
struct SourceLocation;
|
struct SourceLocation;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
struct AsmAnalysisInfo;
|
struct AsmAnalysisInfo;
|
||||||
@ -61,10 +57,10 @@ public:
|
|||||||
explicit AsmAnalyzer(
|
explicit AsmAnalyzer(
|
||||||
AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
langutil::ErrorReporter& _errorReporter,
|
langutil::ErrorReporter& _errorReporter,
|
||||||
EVMVersion _evmVersion,
|
dev::solidity::EVMVersion _evmVersion,
|
||||||
boost::optional<langutil::Error::Type> _errorTypeForLoose,
|
boost::optional<langutil::Error::Type> _errorTypeForLoose,
|
||||||
AsmFlavour _flavour = AsmFlavour::Loose,
|
AsmFlavour _flavour = AsmFlavour::Loose,
|
||||||
yul::ExternalIdentifierAccess::Resolver const& _resolver = yul::ExternalIdentifierAccess::Resolver()
|
ExternalIdentifierAccess::Resolver const& _resolver = ExternalIdentifierAccess::Resolver()
|
||||||
):
|
):
|
||||||
m_resolver(_resolver),
|
m_resolver(_resolver),
|
||||||
m_info(_analysisInfo),
|
m_info(_analysisInfo),
|
||||||
@ -74,23 +70,23 @@ public:
|
|||||||
m_errorTypeForLoose(_errorTypeForLoose)
|
m_errorTypeForLoose(_errorTypeForLoose)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
bool analyze(assembly::Block const& _block);
|
bool analyze(Block const& _block);
|
||||||
|
|
||||||
bool operator()(assembly::Instruction const&);
|
bool operator()(Instruction const&);
|
||||||
bool operator()(assembly::Literal const& _literal);
|
bool operator()(Literal const& _literal);
|
||||||
bool operator()(assembly::Identifier const&);
|
bool operator()(Identifier const&);
|
||||||
bool operator()(assembly::FunctionalInstruction const& _functionalInstruction);
|
bool operator()(FunctionalInstruction const& _functionalInstruction);
|
||||||
bool operator()(assembly::Label const& _label);
|
bool operator()(Label const& _label);
|
||||||
bool operator()(assembly::ExpressionStatement const&);
|
bool operator()(ExpressionStatement const&);
|
||||||
bool operator()(assembly::StackAssignment const&);
|
bool operator()(StackAssignment const&);
|
||||||
bool operator()(assembly::Assignment const& _assignment);
|
bool operator()(Assignment const& _assignment);
|
||||||
bool operator()(assembly::VariableDeclaration const& _variableDeclaration);
|
bool operator()(VariableDeclaration const& _variableDeclaration);
|
||||||
bool operator()(assembly::FunctionDefinition const& _functionDefinition);
|
bool operator()(FunctionDefinition const& _functionDefinition);
|
||||||
bool operator()(assembly::FunctionCall const& _functionCall);
|
bool operator()(FunctionCall const& _functionCall);
|
||||||
bool operator()(assembly::If const& _if);
|
bool operator()(If const& _if);
|
||||||
bool operator()(assembly::Switch const& _switch);
|
bool operator()(Switch const& _switch);
|
||||||
bool operator()(assembly::ForLoop const& _forLoop);
|
bool operator()(ForLoop const& _forLoop);
|
||||||
bool operator()(assembly::Block const& _block);
|
bool operator()(Block const& _block);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// Visits the statement and expects it to deposit one item onto the stack.
|
/// Visits the statement and expects it to deposit one item onto the stack.
|
||||||
@ -99,11 +95,11 @@ private:
|
|||||||
|
|
||||||
/// Verifies that a variable to be assigned to exists and has the same size
|
/// Verifies that a variable to be assigned to exists and has the same size
|
||||||
/// as the value, @a _valueSize, unless that is equal to -1.
|
/// as the value, @a _valueSize, unless that is equal to -1.
|
||||||
bool checkAssignment(assembly::Identifier const& _assignment, size_t _valueSize = size_t(-1));
|
bool checkAssignment(Identifier const& _assignment, size_t _valueSize = size_t(-1));
|
||||||
|
|
||||||
Scope& scope(assembly::Block const* _block);
|
Scope& scope(Block const* _block);
|
||||||
void expectValidType(std::string const& type, langutil::SourceLocation const& _location);
|
void expectValidType(std::string const& type, langutil::SourceLocation const& _location);
|
||||||
void warnOnInstructions(solidity::Instruction _instr, langutil::SourceLocation const& _location);
|
void warnOnInstructions(dev::solidity::Instruction _instr, langutil::SourceLocation const& _location);
|
||||||
|
|
||||||
/// Depending on @a m_flavour and @a m_errorTypeForLoose, throws an internal compiler
|
/// Depending on @a m_flavour and @a m_errorTypeForLoose, throws an internal compiler
|
||||||
/// exception (if the flavour is not Loose), reports an error/warning
|
/// exception (if the flavour is not Loose), reports an error/warning
|
||||||
@ -118,11 +114,9 @@ private:
|
|||||||
std::set<Scope::Variable const*> m_activeVariables;
|
std::set<Scope::Variable const*> m_activeVariables;
|
||||||
AsmAnalysisInfo& m_info;
|
AsmAnalysisInfo& m_info;
|
||||||
langutil::ErrorReporter& m_errorReporter;
|
langutil::ErrorReporter& m_errorReporter;
|
||||||
EVMVersion m_evmVersion;
|
dev::solidity::EVMVersion m_evmVersion;
|
||||||
AsmFlavour m_flavour = AsmFlavour::Loose;
|
AsmFlavour m_flavour = AsmFlavour::Loose;
|
||||||
boost::optional<langutil::Error::Type> m_errorTypeForLoose;
|
boost::optional<langutil::Error::Type> m_errorTypeForLoose;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -18,9 +18,9 @@
|
|||||||
* Information generated during analyzer part of inline assembly.
|
* Information generated during analyzer part of inline assembly.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
|
|
||||||
#include <ostream>
|
#include <ostream>
|
||||||
|
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
|
|
||||||
@ -28,11 +28,7 @@
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
struct Scope;
|
struct Scope;
|
||||||
@ -40,13 +36,11 @@ struct Scope;
|
|||||||
struct AsmAnalysisInfo
|
struct AsmAnalysisInfo
|
||||||
{
|
{
|
||||||
using StackHeightInfo = std::map<void const*, int>;
|
using StackHeightInfo = std::map<void const*, int>;
|
||||||
using Scopes = std::map<assembly::Block const*, std::shared_ptr<Scope>>;
|
using Scopes = std::map<Block const*, std::shared_ptr<Scope>>;
|
||||||
Scopes scopes;
|
Scopes scopes;
|
||||||
StackHeightInfo stackHeightInfo;
|
StackHeightInfo stackHeightInfo;
|
||||||
/// Virtual blocks which will be used for scopes for function arguments and return values.
|
/// Virtual blocks which will be used for scopes for function arguments and return values.
|
||||||
std::map<FunctionDefinition const*, std::shared_ptr<assembly::Block const>> virtualBlocks;
|
std::map<FunctionDefinition const*, std::shared_ptr<Block const>> virtualBlocks;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -20,21 +20,22 @@
|
|||||||
* Code-generating part of inline assembly.
|
* Code-generating part of inline assembly.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmCodeGen.h>
|
#include <libyul/AsmCodeGen.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
|
||||||
#include <libevmasm/Assembly.h>
|
|
||||||
#include <liblangutil/SourceLocation.h>
|
|
||||||
#include <libevmasm/Instruction.h>
|
|
||||||
|
|
||||||
#include <libyul/backends/evm/AbstractAssembly.h>
|
#include <libyul/backends/evm/AbstractAssembly.h>
|
||||||
#include <libyul/backends/evm/EVMCodeTransform.h>
|
#include <libyul/backends/evm/EVMCodeTransform.h>
|
||||||
|
|
||||||
|
#include <libevmasm/Assembly.h>
|
||||||
|
#include <libevmasm/Instruction.h>
|
||||||
|
|
||||||
|
#include <liblangutil/SourceLocation.h>
|
||||||
|
|
||||||
#include <libdevcore/CommonIO.h>
|
#include <libdevcore/CommonIO.h>
|
||||||
|
|
||||||
#include <boost/range/adaptor/reversed.hpp>
|
#include <boost/range/adaptor/reversed.hpp>
|
||||||
@ -47,10 +48,10 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
using namespace dev::solidity::assembly;
|
|
||||||
|
|
||||||
class EthAssemblyAdapter: public yul::AbstractAssembly
|
class EthAssemblyAdapter: public AbstractAssembly
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
explicit EthAssemblyAdapter(eth::Assembly& _assembly):
|
explicit EthAssemblyAdapter(eth::Assembly& _assembly):
|
||||||
@ -142,16 +143,16 @@ private:
|
|||||||
eth::Assembly& m_assembly;
|
eth::Assembly& m_assembly;
|
||||||
};
|
};
|
||||||
|
|
||||||
void assembly::CodeGenerator::assemble(
|
void CodeGenerator::assemble(
|
||||||
Block const& _parsedData,
|
Block const& _parsedData,
|
||||||
AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
eth::Assembly& _assembly,
|
eth::Assembly& _assembly,
|
||||||
yul::ExternalIdentifierAccess const& _identifierAccess,
|
ExternalIdentifierAccess const& _identifierAccess,
|
||||||
bool _useNamedLabelsForFunctions
|
bool _useNamedLabelsForFunctions
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
EthAssemblyAdapter assemblyAdapter(_assembly);
|
EthAssemblyAdapter assemblyAdapter(_assembly);
|
||||||
yul::CodeTransform(
|
CodeTransform(
|
||||||
assemblyAdapter,
|
assemblyAdapter,
|
||||||
_analysisInfo,
|
_analysisInfo,
|
||||||
false,
|
false,
|
@ -22,7 +22,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
|
||||||
@ -32,9 +32,9 @@ namespace eth
|
|||||||
{
|
{
|
||||||
class Assembly;
|
class Assembly;
|
||||||
}
|
}
|
||||||
namespace solidity
|
}
|
||||||
{
|
|
||||||
namespace assembly
|
namespace yul
|
||||||
{
|
{
|
||||||
struct Block;
|
struct Block;
|
||||||
|
|
||||||
@ -45,12 +45,10 @@ public:
|
|||||||
static void assemble(
|
static void assemble(
|
||||||
Block const& _parsedData,
|
Block const& _parsedData,
|
||||||
AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
eth::Assembly& _assembly,
|
dev::eth::Assembly& _assembly,
|
||||||
yul::ExternalIdentifierAccess const& _identifierAccess = yul::ExternalIdentifierAccess(),
|
yul::ExternalIdentifierAccess const& _identifierAccess = yul::ExternalIdentifierAccess(),
|
||||||
bool _useNamedLabelsForFunctions = false
|
bool _useNamedLabelsForFunctions = false
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -22,34 +22,28 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <libevmasm/Instruction.h>
|
#include <libevmasm/Instruction.h>
|
||||||
#include <liblangutil/SourceLocation.h>
|
#include <liblangutil/SourceLocation.h>
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
#include <boost/noncopyable.hpp>
|
#include <boost/noncopyable.hpp>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
using YulString = dev::yul::YulString;
|
|
||||||
using Type = YulString;
|
using Type = YulString;
|
||||||
|
|
||||||
struct TypedName { langutil::SourceLocation location; YulString name; Type type; };
|
struct TypedName { langutil::SourceLocation location; YulString name; Type type; };
|
||||||
using TypedNameList = std::vector<TypedName>;
|
using TypedNameList = std::vector<TypedName>;
|
||||||
|
|
||||||
/// Direct EVM instruction (except PUSHi and JUMPDEST)
|
/// Direct EVM instruction (except PUSHi and JUMPDEST)
|
||||||
struct Instruction { langutil::SourceLocation location; solidity::Instruction instruction; };
|
struct Instruction { langutil::SourceLocation location; dev::solidity::Instruction instruction; };
|
||||||
/// 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 { langutil::SourceLocation location; LiteralKind kind; YulString value; Type type; };
|
||||||
@ -67,7 +61,7 @@ struct StackAssignment { langutil::SourceLocation location; Identifier variableN
|
|||||||
/// 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::shared_ptr<Expression> value; };
|
struct Assignment { langutil::SourceLocation location; std::vector<Identifier> variableNames; std::shared_ptr<Expression> value; };
|
||||||
/// Functional instruction, e.g. "mul(mload(20:u256), add(2:u256, x))"
|
/// Functional instruction, e.g. "mul(mload(20:u256), add(2:u256, x))"
|
||||||
struct FunctionalInstruction { langutil::SourceLocation location; solidity::Instruction instruction; std::vector<Expression> arguments; };
|
struct FunctionalInstruction { langutil::SourceLocation location; dev::solidity::Instruction instruction; std::vector<Expression> arguments; };
|
||||||
struct FunctionCall { langutil::SourceLocation location; Identifier functionName; std::vector<Expression> arguments; };
|
struct FunctionCall { langutil::SourceLocation location; 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 { langutil::SourceLocation location; Expression expression; };
|
||||||
@ -100,5 +94,3 @@ template <class T> inline langutil::SourceLocation locationOf(T const& _node)
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -24,11 +24,7 @@
|
|||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
struct Instruction;
|
struct Instruction;
|
||||||
@ -61,5 +57,3 @@ enum class AsmFlavour
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -20,7 +20,7 @@
|
|||||||
* Solidity inline assembly parser.
|
* Solidity inline assembly parser.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmParser.h>
|
#include <libyul/AsmParser.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
|
|
||||||
@ -32,10 +32,10 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
using namespace dev::solidity::assembly;
|
|
||||||
|
|
||||||
shared_ptr<assembly::Block> Parser::parse(std::shared_ptr<Scanner> const& _scanner, bool _reuseScanner)
|
shared_ptr<Block> Parser::parse(std::shared_ptr<Scanner> const& _scanner, bool _reuseScanner)
|
||||||
{
|
{
|
||||||
m_recursionDepth = 0;
|
m_recursionDepth = 0;
|
||||||
try
|
try
|
||||||
@ -54,10 +54,10 @@ shared_ptr<assembly::Block> Parser::parse(std::shared_ptr<Scanner> const& _scann
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Block Parser::parseBlock()
|
Block Parser::parseBlock()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
assembly::Block block = createWithLocation<Block>();
|
Block block = createWithLocation<Block>();
|
||||||
expectToken(Token::LBrace);
|
expectToken(Token::LBrace);
|
||||||
while (currentToken() != Token::RBrace)
|
while (currentToken() != Token::RBrace)
|
||||||
block.statements.emplace_back(parseStatement());
|
block.statements.emplace_back(parseStatement());
|
||||||
@ -66,7 +66,7 @@ assembly::Block Parser::parseBlock()
|
|||||||
return block;
|
return block;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Statement Parser::parseStatement()
|
Statement Parser::parseStatement()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
switch (currentToken())
|
switch (currentToken())
|
||||||
@ -79,7 +79,7 @@ assembly::Statement Parser::parseStatement()
|
|||||||
return parseBlock();
|
return parseBlock();
|
||||||
case Token::If:
|
case Token::If:
|
||||||
{
|
{
|
||||||
assembly::If _if = createWithLocation<assembly::If>();
|
If _if = createWithLocation<If>();
|
||||||
m_scanner->next();
|
m_scanner->next();
|
||||||
_if.condition = make_shared<Expression>(parseExpression());
|
_if.condition = make_shared<Expression>(parseExpression());
|
||||||
_if.body = parseBlock();
|
_if.body = parseBlock();
|
||||||
@ -87,7 +87,7 @@ assembly::Statement Parser::parseStatement()
|
|||||||
}
|
}
|
||||||
case Token::Switch:
|
case Token::Switch:
|
||||||
{
|
{
|
||||||
assembly::Switch _switch = createWithLocation<assembly::Switch>();
|
Switch _switch = createWithLocation<Switch>();
|
||||||
m_scanner->next();
|
m_scanner->next();
|
||||||
_switch.expression = make_shared<Expression>(parseExpression());
|
_switch.expression = make_shared<Expression>(parseExpression());
|
||||||
while (m_scanner->currentToken() == Token::Case)
|
while (m_scanner->currentToken() == Token::Case)
|
||||||
@ -109,7 +109,7 @@ assembly::Statement Parser::parseStatement()
|
|||||||
{
|
{
|
||||||
if (m_flavour != AsmFlavour::Loose)
|
if (m_flavour != AsmFlavour::Loose)
|
||||||
break;
|
break;
|
||||||
assembly::StackAssignment assignment = createWithLocation<assembly::StackAssignment>();
|
StackAssignment assignment = createWithLocation<StackAssignment>();
|
||||||
advance();
|
advance();
|
||||||
expectToken(Token::Colon);
|
expectToken(Token::Colon);
|
||||||
assignment.variableName.location = location();
|
assignment.variableName.location = location();
|
||||||
@ -139,9 +139,9 @@ assembly::Statement Parser::parseStatement()
|
|||||||
{
|
{
|
||||||
// if a comma follows, a multiple assignment is assumed
|
// if a comma follows, a multiple assignment is assumed
|
||||||
|
|
||||||
if (elementary.type() != typeid(assembly::Identifier))
|
if (elementary.type() != typeid(Identifier))
|
||||||
fatalParserError("Label name / variable name must precede \",\" (multiple assignment).");
|
fatalParserError("Label name / variable name must precede \",\" (multiple assignment).");
|
||||||
assembly::Identifier const& identifier = boost::get<assembly::Identifier>(elementary);
|
Identifier const& identifier = boost::get<Identifier>(elementary);
|
||||||
|
|
||||||
Assignment assignment = createWithLocation<Assignment>(identifier.location);
|
Assignment assignment = createWithLocation<Assignment>(identifier.location);
|
||||||
assignment.variableNames.emplace_back(identifier);
|
assignment.variableNames.emplace_back(identifier);
|
||||||
@ -150,9 +150,9 @@ assembly::Statement Parser::parseStatement()
|
|||||||
{
|
{
|
||||||
expectToken(Token::Comma);
|
expectToken(Token::Comma);
|
||||||
elementary = parseElementaryOperation();
|
elementary = parseElementaryOperation();
|
||||||
if (elementary.type() != typeid(assembly::Identifier))
|
if (elementary.type() != typeid(Identifier))
|
||||||
fatalParserError("Variable name expected in multiple assignment.");
|
fatalParserError("Variable name expected in multiple assignment.");
|
||||||
assignment.variableNames.emplace_back(boost::get<assembly::Identifier>(elementary));
|
assignment.variableNames.emplace_back(boost::get<Identifier>(elementary));
|
||||||
}
|
}
|
||||||
while (currentToken() == Token::Comma);
|
while (currentToken() == Token::Comma);
|
||||||
|
|
||||||
@ -165,15 +165,15 @@ assembly::Statement Parser::parseStatement()
|
|||||||
}
|
}
|
||||||
case Token::Colon:
|
case Token::Colon:
|
||||||
{
|
{
|
||||||
if (elementary.type() != typeid(assembly::Identifier))
|
if (elementary.type() != typeid(Identifier))
|
||||||
fatalParserError("Label name / variable name must precede \":\".");
|
fatalParserError("Label name / variable name must precede \":\".");
|
||||||
assembly::Identifier const& identifier = boost::get<assembly::Identifier>(elementary);
|
Identifier const& identifier = boost::get<Identifier>(elementary);
|
||||||
advance();
|
advance();
|
||||||
// identifier:=: should be parsed as identifier: =: (i.e. a label),
|
// identifier:=: should be parsed as identifier: =: (i.e. a label),
|
||||||
// while identifier:= (being followed by a non-colon) as identifier := (assignment).
|
// while identifier:= (being followed by a non-colon) as identifier := (assignment).
|
||||||
if (currentToken() == Token::Assign && peekNextToken() != Token::Colon)
|
if (currentToken() == Token::Assign && peekNextToken() != Token::Colon)
|
||||||
{
|
{
|
||||||
assembly::Assignment assignment = createWithLocation<assembly::Assignment>(identifier.location);
|
Assignment assignment = createWithLocation<Assignment>(identifier.location);
|
||||||
if (m_flavour != AsmFlavour::Yul && instructions().count(identifier.name.str()))
|
if (m_flavour != AsmFlavour::Yul && instructions().count(identifier.name.str()))
|
||||||
fatalParserError("Cannot use instruction names for identifier names.");
|
fatalParserError("Cannot use instruction names for identifier names.");
|
||||||
advance();
|
advance();
|
||||||
@ -197,36 +197,36 @@ assembly::Statement Parser::parseStatement()
|
|||||||
fatalParserError("Call or assignment expected.");
|
fatalParserError("Call or assignment expected.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (elementary.type() == typeid(assembly::Identifier))
|
if (elementary.type() == typeid(Identifier))
|
||||||
{
|
{
|
||||||
Expression expr = boost::get<assembly::Identifier>(elementary);
|
Expression expr = boost::get<Identifier>(elementary);
|
||||||
return ExpressionStatement{locationOf(expr), expr};
|
return ExpressionStatement{locationOf(expr), expr};
|
||||||
}
|
}
|
||||||
else if (elementary.type() == typeid(assembly::Literal))
|
else if (elementary.type() == typeid(Literal))
|
||||||
{
|
{
|
||||||
Expression expr = boost::get<assembly::Literal>(elementary);
|
Expression expr = boost::get<Literal>(elementary);
|
||||||
return ExpressionStatement{locationOf(expr), expr};
|
return ExpressionStatement{locationOf(expr), expr};
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(elementary.type() == typeid(assembly::Instruction), "Invalid elementary operation.");
|
solAssert(elementary.type() == typeid(Instruction), "Invalid elementary operation.");
|
||||||
return boost::get<assembly::Instruction>(elementary);
|
return boost::get<Instruction>(elementary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Case Parser::parseCase()
|
Case Parser::parseCase()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
assembly::Case _case = createWithLocation<assembly::Case>();
|
Case _case = createWithLocation<Case>();
|
||||||
if (m_scanner->currentToken() == Token::Default)
|
if (m_scanner->currentToken() == Token::Default)
|
||||||
m_scanner->next();
|
m_scanner->next();
|
||||||
else if (m_scanner->currentToken() == Token::Case)
|
else if (m_scanner->currentToken() == Token::Case)
|
||||||
{
|
{
|
||||||
m_scanner->next();
|
m_scanner->next();
|
||||||
ElementaryOperation literal = parseElementaryOperation();
|
ElementaryOperation literal = parseElementaryOperation();
|
||||||
if (literal.type() != typeid(assembly::Literal))
|
if (literal.type() != typeid(Literal))
|
||||||
fatalParserError("Literal expected.");
|
fatalParserError("Literal expected.");
|
||||||
_case.value = make_shared<Literal>(boost::get<assembly::Literal>(std::move(literal)));
|
_case.value = make_shared<Literal>(boost::get<Literal>(std::move(literal)));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
fatalParserError("Case or default case expected.");
|
fatalParserError("Case or default case expected.");
|
||||||
@ -235,7 +235,7 @@ assembly::Case Parser::parseCase()
|
|||||||
return _case;
|
return _case;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::ForLoop Parser::parseForLoop()
|
ForLoop Parser::parseForLoop()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
ForLoop forLoop = createWithLocation<ForLoop>();
|
ForLoop forLoop = createWithLocation<ForLoop>();
|
||||||
@ -248,7 +248,7 @@ assembly::ForLoop Parser::parseForLoop()
|
|||||||
return forLoop;
|
return forLoop;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Expression Parser::parseExpression()
|
Expression Parser::parseExpression()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
// In strict mode, this might parse a plain Instruction, but
|
// In strict mode, this might parse a plain Instruction, but
|
||||||
@ -293,12 +293,12 @@ assembly::Expression Parser::parseExpression()
|
|||||||
Instruction& instr = boost::get<Instruction>(operation);
|
Instruction& instr = boost::get<Instruction>(operation);
|
||||||
return FunctionalInstruction{std::move(instr.location), instr.instruction, {}};
|
return FunctionalInstruction{std::move(instr.location), instr.instruction, {}};
|
||||||
}
|
}
|
||||||
else if (operation.type() == typeid(assembly::Identifier))
|
else if (operation.type() == typeid(Identifier))
|
||||||
return boost::get<assembly::Identifier>(operation);
|
return boost::get<Identifier>(operation);
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
solAssert(operation.type() == typeid(assembly::Literal), "");
|
solAssert(operation.type() == typeid(Literal), "");
|
||||||
return boost::get<assembly::Literal>(operation);
|
return boost::get<Literal>(operation);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -420,7 +420,7 @@ Parser::ElementaryOperation Parser::parseElementaryOperation()
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::VariableDeclaration Parser::parseVariableDeclaration()
|
VariableDeclaration Parser::parseVariableDeclaration()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
VariableDeclaration varDecl = createWithLocation<VariableDeclaration>();
|
VariableDeclaration varDecl = createWithLocation<VariableDeclaration>();
|
||||||
@ -445,7 +445,7 @@ assembly::VariableDeclaration Parser::parseVariableDeclaration()
|
|||||||
return varDecl;
|
return varDecl;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::FunctionDefinition Parser::parseFunctionDefinition()
|
FunctionDefinition Parser::parseFunctionDefinition()
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
FunctionDefinition funDef = createWithLocation<FunctionDefinition>();
|
FunctionDefinition funDef = createWithLocation<FunctionDefinition>();
|
||||||
@ -477,7 +477,7 @@ assembly::FunctionDefinition Parser::parseFunctionDefinition()
|
|||||||
return funDef;
|
return funDef;
|
||||||
}
|
}
|
||||||
|
|
||||||
assembly::Expression Parser::parseCall(Parser::ElementaryOperation&& _initialOp)
|
Expression Parser::parseCall(Parser::ElementaryOperation&& _initialOp)
|
||||||
{
|
{
|
||||||
RecursionGuard recursionGuard(*this);
|
RecursionGuard recursionGuard(*this);
|
||||||
if (_initialOp.type() == typeid(Instruction))
|
if (_initialOp.type() == typeid(Instruction))
|
@ -24,16 +24,12 @@
|
|||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <liblangutil/SourceLocation.h>
|
#include <liblangutil/SourceLocation.h>
|
||||||
#include <liblangutil/Scanner.h>
|
#include <liblangutil/Scanner.h>
|
||||||
#include <liblangutil/ParserBase.h>
|
#include <liblangutil/ParserBase.h>
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
class Parser: public langutil::ParserBase
|
class Parser: public langutil::ParserBase
|
||||||
@ -48,7 +44,7 @@ public:
|
|||||||
std::shared_ptr<Block> parse(std::shared_ptr<langutil::Scanner> const& _scanner, bool _reuseScanner);
|
std::shared_ptr<Block> parse(std::shared_ptr<langutil::Scanner> const& _scanner, bool _reuseScanner);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
using ElementaryOperation = boost::variant<assembly::Instruction, assembly::Literal, assembly::Identifier>;
|
using ElementaryOperation = boost::variant<Instruction, Literal, Identifier>;
|
||||||
|
|
||||||
/// Creates an inline assembly node with the given source location.
|
/// Creates an inline assembly node with the given source location.
|
||||||
template <class T> T createWithLocation(langutil::SourceLocation const& _loc = {}) const
|
template <class T> T createWithLocation(langutil::SourceLocation const& _loc = {}) const
|
||||||
@ -71,7 +67,7 @@ protected:
|
|||||||
Case parseCase();
|
Case parseCase();
|
||||||
ForLoop parseForLoop();
|
ForLoop parseForLoop();
|
||||||
/// Parses a functional expression that has to push exactly one stack element
|
/// Parses a functional expression that has to push exactly one stack element
|
||||||
assembly::Expression parseExpression();
|
Expression parseExpression();
|
||||||
static std::map<std::string, dev::solidity::Instruction> const& instructions();
|
static std::map<std::string, dev::solidity::Instruction> const& instructions();
|
||||||
static std::map<dev::solidity::Instruction, std::string> const& instructionNames();
|
static std::map<dev::solidity::Instruction, std::string> const& instructionNames();
|
||||||
/// Parses an elementary operation, i.e. a literal, identifier or instruction.
|
/// Parses an elementary operation, i.e. a literal, identifier or instruction.
|
||||||
@ -80,7 +76,7 @@ protected:
|
|||||||
ElementaryOperation parseElementaryOperation();
|
ElementaryOperation parseElementaryOperation();
|
||||||
VariableDeclaration parseVariableDeclaration();
|
VariableDeclaration parseVariableDeclaration();
|
||||||
FunctionDefinition parseFunctionDefinition();
|
FunctionDefinition parseFunctionDefinition();
|
||||||
assembly::Expression parseCall(ElementaryOperation&& _initialOp);
|
Expression parseCall(ElementaryOperation&& _initialOp);
|
||||||
TypedName parseTypedName();
|
TypedName parseTypedName();
|
||||||
std::string expectAsmIdentifier();
|
std::string expectAsmIdentifier();
|
||||||
|
|
||||||
@ -91,5 +87,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -20,8 +20,8 @@
|
|||||||
* Converts a parsed assembly into its textual form.
|
* Converts a parsed assembly into its textual form.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmPrinter.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
@ -35,19 +35,19 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
using namespace dev::solidity::assembly;
|
|
||||||
|
|
||||||
//@TODO source locations
|
//@TODO source locations
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::Instruction const& _instruction)
|
string AsmPrinter::operator()(yul::Instruction const& _instruction)
|
||||||
{
|
{
|
||||||
solAssert(!m_yul, "");
|
solAssert(!m_yul, "");
|
||||||
solAssert(isValidInstruction(_instruction.instruction), "Invalid instruction");
|
solAssert(isValidInstruction(_instruction.instruction), "Invalid instruction");
|
||||||
return boost::to_lower_copy(instructionInfo(_instruction.instruction).name);
|
return boost::to_lower_copy(instructionInfo(_instruction.instruction).name);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::Literal const& _literal)
|
string AsmPrinter::operator()(Literal const& _literal)
|
||||||
{
|
{
|
||||||
switch (_literal.kind)
|
switch (_literal.kind)
|
||||||
{
|
{
|
||||||
@ -90,13 +90,13 @@ string AsmPrinter::operator()(assembly::Literal const& _literal)
|
|||||||
return "\"" + out + "\"" + appendTypeName(_literal.type);
|
return "\"" + out + "\"" + appendTypeName(_literal.type);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::Identifier const& _identifier)
|
string AsmPrinter::operator()(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
solAssert(!_identifier.name.empty(), "Invalid identifier.");
|
solAssert(!_identifier.name.empty(), "Invalid identifier.");
|
||||||
return _identifier.name.str();
|
return _identifier.name.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::FunctionalInstruction const& _functionalInstruction)
|
string AsmPrinter::operator()(FunctionalInstruction const& _functionalInstruction)
|
||||||
{
|
{
|
||||||
solAssert(!m_yul, "");
|
solAssert(!m_yul, "");
|
||||||
solAssert(isValidInstruction(_functionalInstruction.instruction), "Invalid instruction");
|
solAssert(isValidInstruction(_functionalInstruction.instruction), "Invalid instruction");
|
||||||
@ -114,21 +114,21 @@ string AsmPrinter::operator()(ExpressionStatement const& _statement)
|
|||||||
return boost::apply_visitor(*this, _statement.expression);
|
return boost::apply_visitor(*this, _statement.expression);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::Label const& _label)
|
string AsmPrinter::operator()(Label const& _label)
|
||||||
{
|
{
|
||||||
solAssert(!m_yul, "");
|
solAssert(!m_yul, "");
|
||||||
solAssert(!_label.name.empty(), "Invalid label.");
|
solAssert(!_label.name.empty(), "Invalid label.");
|
||||||
return _label.name.str() + ":";
|
return _label.name.str() + ":";
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::StackAssignment const& _assignment)
|
string AsmPrinter::operator()(StackAssignment const& _assignment)
|
||||||
{
|
{
|
||||||
solAssert(!m_yul, "");
|
solAssert(!m_yul, "");
|
||||||
solAssert(!_assignment.variableName.name.empty(), "Invalid variable name.");
|
solAssert(!_assignment.variableName.name.empty(), "Invalid variable name.");
|
||||||
return "=: " + (*this)(_assignment.variableName);
|
return "=: " + (*this)(_assignment.variableName);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::Assignment const& _assignment)
|
string AsmPrinter::operator()(Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
solAssert(_assignment.variableNames.size() >= 1, "");
|
solAssert(_assignment.variableNames.size() >= 1, "");
|
||||||
string variables = (*this)(_assignment.variableNames.front());
|
string variables = (*this)(_assignment.variableNames.front());
|
||||||
@ -137,7 +137,7 @@ string AsmPrinter::operator()(assembly::Assignment const& _assignment)
|
|||||||
return variables + " := " + boost::apply_visitor(*this, *_assignment.value);
|
return variables + " := " + boost::apply_visitor(*this, *_assignment.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::VariableDeclaration const& _variableDeclaration)
|
string AsmPrinter::operator()(VariableDeclaration const& _variableDeclaration)
|
||||||
{
|
{
|
||||||
string out = "let ";
|
string out = "let ";
|
||||||
out += boost::algorithm::join(
|
out += boost::algorithm::join(
|
||||||
@ -154,7 +154,7 @@ string AsmPrinter::operator()(assembly::VariableDeclaration const& _variableDecl
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::FunctionDefinition const& _functionDefinition)
|
string AsmPrinter::operator()(FunctionDefinition const& _functionDefinition)
|
||||||
{
|
{
|
||||||
solAssert(!_functionDefinition.name.empty(), "Invalid function name.");
|
solAssert(!_functionDefinition.name.empty(), "Invalid function name.");
|
||||||
string out = "function " + _functionDefinition.name.str() + "(";
|
string out = "function " + _functionDefinition.name.str() + "(";
|
||||||
@ -179,7 +179,7 @@ string AsmPrinter::operator()(assembly::FunctionDefinition const& _functionDefin
|
|||||||
return out + "\n" + (*this)(_functionDefinition.body);
|
return out + "\n" + (*this)(_functionDefinition.body);
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::FunctionCall const& _functionCall)
|
string AsmPrinter::operator()(FunctionCall const& _functionCall)
|
||||||
{
|
{
|
||||||
return
|
return
|
||||||
(*this)(_functionCall.functionName) + "(" +
|
(*this)(_functionCall.functionName) + "(" +
|
||||||
@ -210,7 +210,7 @@ string AsmPrinter::operator()(Switch const& _switch)
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
string AsmPrinter::operator()(assembly::ForLoop const& _forLoop)
|
string AsmPrinter::operator()(ForLoop const& _forLoop)
|
||||||
{
|
{
|
||||||
solAssert(_forLoop.condition, "Invalid for loop condition.");
|
solAssert(_forLoop.condition, "Invalid for loop condition.");
|
||||||
string out = "for ";
|
string out = "for ";
|
62
libyul/AsmPrinter.h
Normal file
62
libyul/AsmPrinter.h
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
/*
|
||||||
|
This file is part of solidity.
|
||||||
|
|
||||||
|
solidity is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
solidity is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* @author Christian <c@ethdev.com>
|
||||||
|
* @date 2017
|
||||||
|
* Converts a parsed assembly into its textual form.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
|
#include <boost/variant.hpp>
|
||||||
|
|
||||||
|
namespace yul
|
||||||
|
{
|
||||||
|
|
||||||
|
class AsmPrinter: public boost::static_visitor<std::string>
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit AsmPrinter(bool _yul = false): m_yul(_yul) {}
|
||||||
|
|
||||||
|
std::string operator()(Instruction const& _instruction);
|
||||||
|
std::string operator()(Literal const& _literal);
|
||||||
|
std::string operator()(Identifier const& _identifier);
|
||||||
|
std::string operator()(FunctionalInstruction const& _functionalInstruction);
|
||||||
|
std::string operator()(ExpressionStatement const& _expr);
|
||||||
|
std::string operator()(Label const& _label);
|
||||||
|
std::string operator()(StackAssignment const& _assignment);
|
||||||
|
std::string operator()(Assignment const& _assignment);
|
||||||
|
std::string operator()(VariableDeclaration const& _variableDeclaration);
|
||||||
|
std::string operator()(FunctionDefinition const& _functionDefinition);
|
||||||
|
std::string operator()(FunctionCall const& _functionCall);
|
||||||
|
std::string operator()(If const& _if);
|
||||||
|
std::string operator()(Switch const& _switch);
|
||||||
|
std::string operator()(ForLoop const& _forLoop);
|
||||||
|
std::string operator()(Block const& _block);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string formatTypedName(TypedName _variable) const;
|
||||||
|
std::string appendTypeName(YulString _type) const;
|
||||||
|
|
||||||
|
bool m_yul = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
@ -18,13 +18,13 @@
|
|||||||
* Scopes for identifiers.
|
* Scopes for identifiers.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::solidity::assembly;
|
using namespace yul;
|
||||||
|
|
||||||
bool Scope::registerLabel(yul::YulString _name)
|
bool Scope::registerLabel(YulString _name)
|
||||||
{
|
{
|
||||||
if (exists(_name))
|
if (exists(_name))
|
||||||
return false;
|
return false;
|
||||||
@ -32,7 +32,7 @@ bool Scope::registerLabel(yul::YulString _name)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Scope::registerVariable(yul::YulString _name, YulType const& _type)
|
bool Scope::registerVariable(YulString _name, YulType const& _type)
|
||||||
{
|
{
|
||||||
if (exists(_name))
|
if (exists(_name))
|
||||||
return false;
|
return false;
|
||||||
@ -42,7 +42,7 @@ bool Scope::registerVariable(yul::YulString _name, YulType const& _type)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Scope::registerFunction(yul::YulString _name, std::vector<YulType> const& _arguments, std::vector<YulType> const& _returns)
|
bool Scope::registerFunction(YulString _name, std::vector<YulType> const& _arguments, std::vector<YulType> const& _returns)
|
||||||
{
|
{
|
||||||
if (exists(_name))
|
if (exists(_name))
|
||||||
return false;
|
return false;
|
||||||
@ -50,7 +50,7 @@ bool Scope::registerFunction(yul::YulString _name, std::vector<YulType> const& _
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Scope::Identifier* Scope::lookup(yul::YulString _name)
|
Scope::Identifier* Scope::lookup(YulString _name)
|
||||||
{
|
{
|
||||||
bool crossedFunctionBoundary = false;
|
bool crossedFunctionBoundary = false;
|
||||||
for (Scope* s = this; s; s = s->superScope)
|
for (Scope* s = this; s; s = s->superScope)
|
||||||
@ -70,7 +70,7 @@ Scope::Identifier* Scope::lookup(yul::YulString _name)
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Scope::exists(yul::YulString _name) const
|
bool Scope::exists(YulString _name) const
|
||||||
{
|
{
|
||||||
if (identifiers.count(_name))
|
if (identifiers.count(_name))
|
||||||
return true;
|
return true;
|
@ -32,16 +32,12 @@
|
|||||||
#include <functional>
|
#include <functional>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
struct Scope
|
struct Scope
|
||||||
{
|
{
|
||||||
using YulType = yul::YulString;
|
using YulType = YulString;
|
||||||
using LabelID = size_t;
|
using LabelID = size_t;
|
||||||
|
|
||||||
struct Variable { YulType type; };
|
struct Variable { YulType type; };
|
||||||
@ -53,13 +49,13 @@ struct Scope
|
|||||||
};
|
};
|
||||||
|
|
||||||
using Identifier = boost::variant<Variable, Label, Function>;
|
using Identifier = boost::variant<Variable, Label, Function>;
|
||||||
using Visitor = GenericVisitor<Variable const, Label const, Function const>;
|
using Visitor = dev::GenericVisitor<Variable const, Label const, Function const>;
|
||||||
using NonconstVisitor = GenericVisitor<Variable, Label, Function>;
|
using NonconstVisitor = dev::GenericVisitor<Variable, Label, Function>;
|
||||||
|
|
||||||
bool registerVariable(yul::YulString _name, YulType const& _type);
|
bool registerVariable(YulString _name, YulType const& _type);
|
||||||
bool registerLabel(yul::YulString _name);
|
bool registerLabel(YulString _name);
|
||||||
bool registerFunction(
|
bool registerFunction(
|
||||||
yul::YulString _name,
|
YulString _name,
|
||||||
std::vector<YulType> const& _arguments,
|
std::vector<YulType> const& _arguments,
|
||||||
std::vector<YulType> const& _returns
|
std::vector<YulType> const& _returns
|
||||||
);
|
);
|
||||||
@ -69,12 +65,12 @@ struct Scope
|
|||||||
/// will any lookups across assembly boundaries.
|
/// will any lookups across assembly boundaries.
|
||||||
/// The pointer will be invalidated if the scope is modified.
|
/// The pointer will be invalidated if the scope is modified.
|
||||||
/// @param _crossedFunction if true, we already crossed a function boundary during recursive lookup
|
/// @param _crossedFunction if true, we already crossed a function boundary during recursive lookup
|
||||||
Identifier* lookup(yul::YulString _name);
|
Identifier* lookup(YulString _name);
|
||||||
/// Looks up the identifier in this and super scopes (will not find variables across function
|
/// Looks up the identifier in this and super scopes (will not find variables across function
|
||||||
/// boundaries and generally stops at assembly boundaries) and calls the visitor, returns
|
/// boundaries and generally stops at assembly boundaries) and calls the visitor, returns
|
||||||
/// false if not found.
|
/// false if not found.
|
||||||
template <class V>
|
template <class V>
|
||||||
bool lookup(yul::YulString _name, V const& _visitor)
|
bool lookup(YulString _name, V const& _visitor)
|
||||||
{
|
{
|
||||||
if (Identifier* id = lookup(_name))
|
if (Identifier* id = lookup(_name))
|
||||||
{
|
{
|
||||||
@ -86,7 +82,7 @@ struct Scope
|
|||||||
}
|
}
|
||||||
/// @returns true if the name exists in this scope or in super scopes (also searches
|
/// @returns true if the name exists in this scope or in super scopes (also searches
|
||||||
/// across function and assembly boundaries).
|
/// across function and assembly boundaries).
|
||||||
bool exists(yul::YulString _name) const;
|
bool exists(YulString _name) const;
|
||||||
|
|
||||||
/// @returns the number of variables directly registered inside the scope.
|
/// @returns the number of variables directly registered inside the scope.
|
||||||
size_t numberOfVariables() const;
|
size_t numberOfVariables() const;
|
||||||
@ -97,9 +93,7 @@ struct Scope
|
|||||||
/// If true, variables from the super scope are not visible here (other identifiers are),
|
/// If true, variables from the super scope are not visible here (other identifiers are),
|
||||||
/// but they are still taken into account to prevent shadowing.
|
/// but they are still taken into account to prevent shadowing.
|
||||||
bool functionScope = false;
|
bool functionScope = false;
|
||||||
std::map<yul::YulString, Identifier> identifiers;
|
std::map<YulString, Identifier> identifiers;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
@ -18,11 +18,11 @@
|
|||||||
* Module responsible for registering identifiers inside their scopes.
|
* Module responsible for registering identifiers inside their scopes.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmScopeFiller.h>
|
#include <libyul/AsmScopeFiller.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
|
|
||||||
#include <liblangutil/ErrorReporter.h>
|
#include <liblangutil/ErrorReporter.h>
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
@ -37,8 +37,8 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
using namespace dev::solidity::assembly;
|
|
||||||
|
|
||||||
ScopeFiller::ScopeFiller(AsmAnalysisInfo& _info, ErrorReporter& _errorReporter):
|
ScopeFiller::ScopeFiller(AsmAnalysisInfo& _info, ErrorReporter& _errorReporter):
|
||||||
m_info(_info), m_errorReporter(_errorReporter)
|
m_info(_info), m_errorReporter(_errorReporter)
|
||||||
@ -65,7 +65,7 @@ bool ScopeFiller::operator()(Label const& _item)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ScopeFiller::operator()(assembly::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.location, *m_currentScope))
|
||||||
@ -73,7 +73,7 @@ bool ScopeFiller::operator()(assembly::VariableDeclaration const& _varDecl)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ScopeFiller::operator()(assembly::FunctionDefinition const& _funDef)
|
bool ScopeFiller::operator()(FunctionDefinition const& _funDef)
|
||||||
{
|
{
|
||||||
bool success = true;
|
bool success = true;
|
||||||
vector<Scope::YulType> arguments;
|
vector<Scope::YulType> arguments;
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
|
|
||||||
@ -33,11 +33,7 @@ class ErrorReporter;
|
|||||||
struct SourceLocation;
|
struct SourceLocation;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace dev
|
namespace yul
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
{
|
||||||
|
|
||||||
struct TypedName;
|
struct TypedName;
|
||||||
@ -53,21 +49,21 @@ class ScopeFiller: public boost::static_visitor<bool>
|
|||||||
public:
|
public:
|
||||||
ScopeFiller(AsmAnalysisInfo& _info, langutil::ErrorReporter& _errorReporter);
|
ScopeFiller(AsmAnalysisInfo& _info, langutil::ErrorReporter& _errorReporter);
|
||||||
|
|
||||||
bool operator()(assembly::Instruction const&) { return true; }
|
bool operator()(Instruction const&) { return true; }
|
||||||
bool operator()(assembly::Literal const&) { return true; }
|
bool operator()(Literal const&) { return true; }
|
||||||
bool operator()(assembly::Identifier const&) { return true; }
|
bool operator()(Identifier const&) { return true; }
|
||||||
bool operator()(assembly::FunctionalInstruction const&) { return true; }
|
bool operator()(FunctionalInstruction const&) { return true; }
|
||||||
bool operator()(assembly::ExpressionStatement const& _expr);
|
bool operator()(ExpressionStatement const& _expr);
|
||||||
bool operator()(assembly::Label const& _label);
|
bool operator()(Label const& _label);
|
||||||
bool operator()(assembly::StackAssignment const&) { return true; }
|
bool operator()(StackAssignment const&) { return true; }
|
||||||
bool operator()(assembly::Assignment const&) { return true; }
|
bool operator()(Assignment const&) { return true; }
|
||||||
bool operator()(assembly::VariableDeclaration const& _variableDeclaration);
|
bool operator()(VariableDeclaration const& _variableDeclaration);
|
||||||
bool operator()(assembly::FunctionDefinition const& _functionDefinition);
|
bool operator()(FunctionDefinition const& _functionDefinition);
|
||||||
bool operator()(assembly::FunctionCall const&) { return true; }
|
bool operator()(FunctionCall const&) { return true; }
|
||||||
bool operator()(assembly::If const& _if);
|
bool operator()(If const& _if);
|
||||||
bool operator()(assembly::Switch const& _switch);
|
bool operator()(Switch const& _switch);
|
||||||
bool operator()(assembly::ForLoop const& _forLoop);
|
bool operator()(ForLoop const& _forLoop);
|
||||||
bool operator()(assembly::Block const& _block);
|
bool operator()(Block const& _block);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool registerVariable(
|
bool registerVariable(
|
||||||
@ -76,7 +72,7 @@ private:
|
|||||||
Scope& _scope
|
Scope& _scope
|
||||||
);
|
);
|
||||||
|
|
||||||
Scope& scope(assembly::Block const* _block);
|
Scope& scope(Block const* _block);
|
||||||
|
|
||||||
Scope* m_currentScope = nullptr;
|
Scope* m_currentScope = nullptr;
|
||||||
AsmAnalysisInfo& m_info;
|
AsmAnalysisInfo& m_info;
|
||||||
@ -84,5 +80,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
43
libyul/CMakeLists.txt
Normal file
43
libyul/CMakeLists.txt
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
add_library(yul
|
||||||
|
AsmAnalysis.cpp
|
||||||
|
AsmAnalysisInfo.cpp
|
||||||
|
AsmCodeGen.cpp
|
||||||
|
AsmParser.cpp
|
||||||
|
AsmPrinter.cpp
|
||||||
|
AsmScope.cpp
|
||||||
|
AsmScopeFiller.cpp
|
||||||
|
backends/evm/EVMAssembly.cpp
|
||||||
|
backends/evm/EVMCodeTransform.cpp
|
||||||
|
optimiser/ASTCopier.cpp
|
||||||
|
optimiser/ASTWalker.cpp
|
||||||
|
optimiser/BlockFlattener.cpp
|
||||||
|
optimiser/CommonSubexpressionEliminator.cpp
|
||||||
|
optimiser/DataFlowAnalyzer.cpp
|
||||||
|
optimiser/Disambiguator.cpp
|
||||||
|
optimiser/ExpressionInliner.cpp
|
||||||
|
optimiser/ExpressionJoiner.cpp
|
||||||
|
optimiser/ExpressionSimplifier.cpp
|
||||||
|
optimiser/ExpressionSplitter.cpp
|
||||||
|
optimiser/ForLoopInitRewriter.cpp
|
||||||
|
optimiser/FullInliner.cpp
|
||||||
|
optimiser/FunctionGrouper.cpp
|
||||||
|
optimiser/FunctionHoister.cpp
|
||||||
|
optimiser/InlinableExpressionFunctionFinder.cpp
|
||||||
|
optimiser/MainFunction.cpp
|
||||||
|
optimiser/Metrics.cpp
|
||||||
|
optimiser/NameCollector.cpp
|
||||||
|
optimiser/NameDispenser.cpp
|
||||||
|
optimiser/RedundantAssignEliminator.cpp
|
||||||
|
optimiser/Rematerialiser.cpp
|
||||||
|
optimiser/SSATransform.cpp
|
||||||
|
optimiser/SSAValueTracker.cpp
|
||||||
|
optimiser/Semantics.cpp
|
||||||
|
optimiser/SimplificationRules.cpp
|
||||||
|
optimiser/Substitution.cpp
|
||||||
|
optimiser/Suite.cpp
|
||||||
|
optimiser/SyntacticalEquality.cpp
|
||||||
|
optimiser/UnusedPruner.cpp
|
||||||
|
optimiser/Utilities.cpp
|
||||||
|
optimiser/VarDeclPropagator.cpp
|
||||||
|
)
|
||||||
|
target_link_libraries(yul PUBLIC devcore)
|
@ -23,18 +23,15 @@
|
|||||||
#include <libdevcore/Exceptions.h>
|
#include <libdevcore/Exceptions.h>
|
||||||
#include <libdevcore/Assertions.h>
|
#include <libdevcore/Assertions.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
struct YulException: virtual Exception {};
|
struct YulException: virtual dev::Exception {};
|
||||||
struct OptimizerException: virtual YulException {};
|
struct OptimizerException: virtual YulException {};
|
||||||
struct YulAssertion: virtual YulException {};
|
struct YulAssertion: virtual YulException {};
|
||||||
|
|
||||||
/// Assertion that throws an YulAssertion containing the given description if it is not met.
|
/// Assertion that throws an YulAssertion containing the given description if it is not met.
|
||||||
#define yulAssert(CONDITION, DESCRIPTION) \
|
#define yulAssert(CONDITION, DESCRIPTION) \
|
||||||
assertThrow(CONDITION, ::dev::yul::YulException, DESCRIPTION)
|
assertThrow(CONDITION, ::yul::YulException, DESCRIPTION)
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -27,8 +27,6 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -130,4 +128,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <libdevcore/Common.h>
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
@ -36,14 +37,13 @@ namespace dev
|
|||||||
namespace solidity
|
namespace solidity
|
||||||
{
|
{
|
||||||
enum class Instruction: uint8_t;
|
enum class Instruction: uint8_t;
|
||||||
namespace assembly
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace yul
|
||||||
{
|
{
|
||||||
struct Instruction;
|
struct Instruction;
|
||||||
struct Identifier;
|
struct Identifier;
|
||||||
}
|
|
||||||
}
|
|
||||||
namespace yul
|
|
||||||
{
|
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Assembly class that abstracts both the libevmasm assembly and the new Yul assembly.
|
/// Assembly class that abstracts both the libevmasm assembly and the new Yul assembly.
|
||||||
@ -61,9 +61,9 @@ public:
|
|||||||
/// at the beginning.
|
/// at the beginning.
|
||||||
virtual int stackHeight() const = 0;
|
virtual int stackHeight() const = 0;
|
||||||
/// Append an EVM instruction.
|
/// Append an EVM instruction.
|
||||||
virtual void appendInstruction(solidity::Instruction _instruction) = 0;
|
virtual void appendInstruction(dev::solidity::Instruction _instruction) = 0;
|
||||||
/// Append a constant.
|
/// Append a constant.
|
||||||
virtual void appendConstant(u256 const& _constant) = 0;
|
virtual void appendConstant(dev::u256 const& _constant) = 0;
|
||||||
/// Append a label.
|
/// Append a label.
|
||||||
virtual void appendLabel(LabelID _labelId) = 0;
|
virtual void appendLabel(LabelID _labelId) = 0;
|
||||||
/// Append a label reference.
|
/// Append a label reference.
|
||||||
@ -106,18 +106,15 @@ enum class IdentifierContext { LValue, RValue };
|
|||||||
/// to inline assembly (not used in standalone assembly mode).
|
/// to inline assembly (not used in standalone assembly mode).
|
||||||
struct ExternalIdentifierAccess
|
struct ExternalIdentifierAccess
|
||||||
{
|
{
|
||||||
using Resolver = std::function<size_t(solidity::assembly::Identifier const&, IdentifierContext, bool /*_crossesFunctionBoundary*/)>;
|
using Resolver = std::function<size_t(Identifier const&, IdentifierContext, bool /*_crossesFunctionBoundary*/)>;
|
||||||
/// Resolve an external reference given by the identifier in the given context.
|
/// Resolve an external reference given by the identifier in the given context.
|
||||||
/// @returns the size of the value (number of stack slots) or size_t(-1) if not found.
|
/// @returns the size of the value (number of stack slots) or size_t(-1) if not found.
|
||||||
Resolver resolve;
|
Resolver resolve;
|
||||||
using CodeGenerator = std::function<void(solidity::assembly::Identifier const&, IdentifierContext, yul::AbstractAssembly&)>;
|
using CodeGenerator = std::function<void(Identifier const&, IdentifierContext, yul::AbstractAssembly&)>;
|
||||||
/// Generate code for retrieving the value (rvalue context) or storing the value (lvalue context)
|
/// Generate code for retrieving the value (rvalue context) or storing the value (lvalue context)
|
||||||
/// of an identifier. The code should be appended to the assembly. In rvalue context, the value is supposed
|
/// of an identifier. The code should be appended to the assembly. In rvalue context, the value is supposed
|
||||||
/// to be put onto the stack, in lvalue context, the value is assumed to be at the top of the stack.
|
/// to be put onto the stack, in lvalue context, the value is assumed to be at the top of the stack.
|
||||||
CodeGenerator generateCode;
|
CodeGenerator generateCode;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,7 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
|
@ -31,8 +31,6 @@ namespace langutil
|
|||||||
struct SourceLocation;
|
struct SourceLocation;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -48,9 +46,9 @@ public:
|
|||||||
/// at the beginning.
|
/// at the beginning.
|
||||||
int stackHeight() const override { return m_stackHeight; }
|
int stackHeight() const override { return m_stackHeight; }
|
||||||
/// Append an EVM instruction.
|
/// Append an EVM instruction.
|
||||||
void appendInstruction(solidity::Instruction _instruction) override;
|
void appendInstruction(dev::solidity::Instruction _instruction) override;
|
||||||
/// Append a constant.
|
/// Append a constant.
|
||||||
void appendConstant(u256 const& _constant) override;
|
void appendConstant(dev::u256 const& _constant) override;
|
||||||
/// Append a label.
|
/// Append a label.
|
||||||
void appendLabel(LabelID _labelId) override;
|
void appendLabel(LabelID _labelId) override;
|
||||||
/// Append a label reference.
|
/// Append a label reference.
|
||||||
@ -81,17 +79,17 @@ public:
|
|||||||
void appendAssemblySize() override;
|
void appendAssemblySize() override;
|
||||||
|
|
||||||
/// Resolves references inside the bytecode and returns the linker object.
|
/// Resolves references inside the bytecode and returns the linker object.
|
||||||
eth::LinkerObject finalize();
|
dev::eth::LinkerObject finalize();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void setLabelToCurrentPosition(AbstractAssembly::LabelID _labelId);
|
void setLabelToCurrentPosition(AbstractAssembly::LabelID _labelId);
|
||||||
void appendLabelReferenceInternal(AbstractAssembly::LabelID _labelId);
|
void appendLabelReferenceInternal(AbstractAssembly::LabelID _labelId);
|
||||||
void updateReference(size_t pos, size_t size, u256 value);
|
void updateReference(size_t pos, size_t size, dev::u256 value);
|
||||||
|
|
||||||
bool m_evm15 = false; ///< if true, switch to evm1.5 mode
|
bool m_evm15 = false; ///< if true, switch to evm1.5 mode
|
||||||
LabelID m_nextLabelId = 0;
|
LabelID m_nextLabelId = 0;
|
||||||
int m_stackHeight = 0;
|
int m_stackHeight = 0;
|
||||||
bytes m_bytecode;
|
dev::bytes m_bytecode;
|
||||||
std::map<std::string, LabelID> m_namedLabels;
|
std::map<std::string, LabelID> m_namedLabels;
|
||||||
std::map<LabelID, size_t> m_labelPositions;
|
std::map<LabelID, size_t> m_labelPositions;
|
||||||
std::map<size_t, LabelID> m_labelReferences;
|
std::map<size_t, LabelID> m_labelReferences;
|
||||||
@ -99,4 +97,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,8 +20,8 @@
|
|||||||
|
|
||||||
#include <libyul/backends/evm/EVMCodeTransform.h>
|
#include <libyul/backends/evm/EVMCodeTransform.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <liblangutil/Exceptions.h>
|
#include <liblangutil/Exceptions.h>
|
||||||
|
|
||||||
@ -29,11 +29,9 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
using Scope = dev::solidity::assembly::Scope;
|
|
||||||
|
|
||||||
void CodeTransform::operator()(VariableDeclaration const& _varDecl)
|
void CodeTransform::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
solAssert(m_scope, "");
|
solAssert(m_scope, "");
|
||||||
@ -147,7 +145,7 @@ void CodeTransform::operator()(FunctionalInstruction const& _instruction)
|
|||||||
solAssert(_instruction.arguments.size() == 1, "");
|
solAssert(_instruction.arguments.size() == 1, "");
|
||||||
}
|
}
|
||||||
m_assembly.setSourceLocation(_instruction.location);
|
m_assembly.setSourceLocation(_instruction.location);
|
||||||
auto label = labelFromIdentifier(boost::get<assembly::Identifier>(_instruction.arguments.at(0)));
|
auto label = labelFromIdentifier(boost::get<Identifier>(_instruction.arguments.at(0)));
|
||||||
if (isJumpI)
|
if (isJumpI)
|
||||||
m_assembly.appendJumpToIf(label);
|
m_assembly.appendJumpToIf(label);
|
||||||
else
|
else
|
||||||
@ -163,7 +161,7 @@ void CodeTransform::operator()(FunctionalInstruction const& _instruction)
|
|||||||
checkStackHeight(&_instruction);
|
checkStackHeight(&_instruction);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::operator()(assembly::Identifier const& _identifier)
|
void CodeTransform::operator()(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_identifier.location);
|
m_assembly.setSourceLocation(_identifier.location);
|
||||||
// First search internals, then externals.
|
// First search internals, then externals.
|
||||||
@ -197,12 +195,12 @@ void CodeTransform::operator()(assembly::Identifier const& _identifier)
|
|||||||
checkStackHeight(&_identifier);
|
checkStackHeight(&_identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::operator()(assembly::Literal const& _literal)
|
void CodeTransform::operator()(Literal const& _literal)
|
||||||
{
|
{
|
||||||
m_assembly.setSourceLocation(_literal.location);
|
m_assembly.setSourceLocation(_literal.location);
|
||||||
if (_literal.kind == assembly::LiteralKind::Number)
|
if (_literal.kind == LiteralKind::Number)
|
||||||
m_assembly.appendConstant(u256(_literal.value.str()));
|
m_assembly.appendConstant(u256(_literal.value.str()));
|
||||||
else if (_literal.kind == assembly::LiteralKind::Boolean)
|
else if (_literal.kind == LiteralKind::Boolean)
|
||||||
{
|
{
|
||||||
if (_literal.value.str() == "true")
|
if (_literal.value.str() == "true")
|
||||||
m_assembly.appendConstant(u256(1));
|
m_assembly.appendConstant(u256(1));
|
||||||
@ -217,7 +215,7 @@ void CodeTransform::operator()(assembly::Literal const& _literal)
|
|||||||
checkStackHeight(&_literal);
|
checkStackHeight(&_literal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeTransform::operator()(assembly::Instruction const& _instruction)
|
void CodeTransform::operator()(yul::Instruction const& _instruction)
|
||||||
{
|
{
|
||||||
solAssert(!m_evm15 || _instruction.instruction != solidity::Instruction::JUMP, "Bare JUMP instruction used for EVM1.5");
|
solAssert(!m_evm15 || _instruction.instruction != solidity::Instruction::JUMP, "Bare JUMP instruction used for EVM1.5");
|
||||||
solAssert(!m_evm15 || _instruction.instruction != solidity::Instruction::JUMPI, "Bare JUMPI instruction used for EVM1.5");
|
solAssert(!m_evm15 || _instruction.instruction != solidity::Instruction::JUMPI, "Bare JUMPI instruction used for EVM1.5");
|
||||||
@ -522,7 +520,7 @@ void CodeTransform::generateAssignment(Identifier const& _variableName)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int CodeTransform::variableHeightDiff(solidity::assembly::Scope::Variable const& _var, bool _forSwap) const
|
int CodeTransform::variableHeightDiff(Scope::Variable const& _var, bool _forSwap) const
|
||||||
{
|
{
|
||||||
solAssert(m_context->variableStackHeights.count(&_var), "");
|
solAssert(m_context->variableStackHeights.count(&_var), "");
|
||||||
int heightDiff = m_assembly.stackHeight() - m_context->variableStackHeights[&_var];
|
int heightDiff = m_assembly.stackHeight() - m_context->variableStackHeights[&_var];
|
||||||
|
@ -20,25 +20,21 @@
|
|||||||
|
|
||||||
#include <libyul/backends/evm/EVMAssembly.h>
|
#include <libyul/backends/evm/EVMAssembly.h>
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
#include <libyul/AsmScope.h>
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
|
|
||||||
namespace dev
|
namespace langutil
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
{
|
||||||
class ErrorReporter;
|
class ErrorReporter;
|
||||||
namespace assembly
|
|
||||||
{
|
|
||||||
struct AsmAnalysisInfo;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
struct AsmAnalysisInfo;
|
||||||
class EVMAssembly;
|
class EVMAssembly;
|
||||||
|
|
||||||
class CodeTransform: public boost::static_visitor<>
|
class CodeTransform: public boost::static_visitor<>
|
||||||
@ -47,8 +43,8 @@ public:
|
|||||||
/// Create the code transformer.
|
/// Create the code transformer.
|
||||||
/// @param _identifierAccess used to resolve identifiers external to the inline assembly
|
/// @param _identifierAccess used to resolve identifiers external to the inline assembly
|
||||||
CodeTransform(
|
CodeTransform(
|
||||||
yul::AbstractAssembly& _assembly,
|
AbstractAssembly& _assembly,
|
||||||
solidity::assembly::AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
bool _yul = false,
|
bool _yul = false,
|
||||||
bool _evm15 = false,
|
bool _evm15 = false,
|
||||||
ExternalIdentifierAccess const& _identifierAccess = ExternalIdentifierAccess(),
|
ExternalIdentifierAccess const& _identifierAccess = ExternalIdentifierAccess(),
|
||||||
@ -69,15 +65,14 @@ public:
|
|||||||
protected:
|
protected:
|
||||||
struct Context
|
struct Context
|
||||||
{
|
{
|
||||||
using Scope = solidity::assembly::Scope;
|
|
||||||
std::map<Scope::Label const*, AbstractAssembly::LabelID> labelIDs;
|
std::map<Scope::Label const*, AbstractAssembly::LabelID> labelIDs;
|
||||||
std::map<Scope::Function const*, AbstractAssembly::LabelID> functionEntryIDs;
|
std::map<Scope::Function const*, AbstractAssembly::LabelID> functionEntryIDs;
|
||||||
std::map<Scope::Variable const*, int> variableStackHeights;
|
std::map<Scope::Variable const*, int> variableStackHeights;
|
||||||
};
|
};
|
||||||
|
|
||||||
CodeTransform(
|
CodeTransform(
|
||||||
yul::AbstractAssembly& _assembly,
|
AbstractAssembly& _assembly,
|
||||||
solidity::assembly::AsmAnalysisInfo& _analysisInfo,
|
AsmAnalysisInfo& _analysisInfo,
|
||||||
bool _yul,
|
bool _yul,
|
||||||
bool _evm15,
|
bool _evm15,
|
||||||
ExternalIdentifierAccess const& _identifierAccess,
|
ExternalIdentifierAccess const& _identifierAccess,
|
||||||
@ -116,8 +111,8 @@ private:
|
|||||||
AbstractAssembly::LabelID labelFromIdentifier(Identifier const& _identifier);
|
AbstractAssembly::LabelID labelFromIdentifier(Identifier const& _identifier);
|
||||||
/// @returns the label ID corresponding to the given label, allocating a new one if
|
/// @returns the label ID corresponding to the given label, allocating a new one if
|
||||||
/// necessary.
|
/// necessary.
|
||||||
AbstractAssembly::LabelID labelID(solidity::assembly::Scope::Label const& _label);
|
AbstractAssembly::LabelID labelID(Scope::Label const& _label);
|
||||||
AbstractAssembly::LabelID functionEntryID(YulString _name, solidity::assembly::Scope::Function const& _function);
|
AbstractAssembly::LabelID functionEntryID(YulString _name, Scope::Function const& _function);
|
||||||
/// Generates code for an expression that is supposed to return a single value.
|
/// Generates code for an expression that is supposed to return a single value.
|
||||||
void visitExpression(Expression const& _expression);
|
void visitExpression(Expression const& _expression);
|
||||||
|
|
||||||
@ -133,15 +128,15 @@ private:
|
|||||||
/// Determines the stack height difference to the given variables. Throws
|
/// Determines the stack height difference to the given variables. Throws
|
||||||
/// if it is not yet in scope or the height difference is too large. Returns
|
/// if it is not yet in scope or the height difference is too large. Returns
|
||||||
/// the (positive) stack height difference otherwise.
|
/// the (positive) stack height difference otherwise.
|
||||||
int variableHeightDiff(solidity::assembly::Scope::Variable const& _var, bool _forSwap) const;
|
int variableHeightDiff(Scope::Variable const& _var, bool _forSwap) const;
|
||||||
|
|
||||||
void expectDeposit(int _deposit, int _oldHeight) const;
|
void expectDeposit(int _deposit, int _oldHeight) const;
|
||||||
|
|
||||||
void checkStackHeight(void const* _astElement) const;
|
void checkStackHeight(void const* _astElement) const;
|
||||||
|
|
||||||
yul::AbstractAssembly& m_assembly;
|
AbstractAssembly& m_assembly;
|
||||||
solidity::assembly::AsmAnalysisInfo& m_info;
|
AsmAnalysisInfo& m_info;
|
||||||
solidity::assembly::Scope* m_scope = nullptr;
|
Scope* m_scope = nullptr;
|
||||||
bool m_yul = false;
|
bool m_yul = false;
|
||||||
bool m_evm15 = false;
|
bool m_evm15 = false;
|
||||||
bool m_useNamedLabelsForFunctions = false;
|
bool m_useNamedLabelsForFunctions = false;
|
||||||
@ -155,4 +150,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -22,13 +22,13 @@
|
|||||||
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <libdevcore/Common.h>
|
#include <libdevcore/Common.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
Statement ASTCopier::operator()(Instruction const&)
|
Statement ASTCopier::operator()(Instruction const&)
|
||||||
{
|
{
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
@ -31,8 +31,6 @@
|
|||||||
#include <set>
|
#include <set>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -123,4 +121,3 @@ std::vector<T> ASTCopier::translateVector(std::vector<T> const& _values)
|
|||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,13 +20,13 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <boost/range/adaptor/reversed.hpp>
|
#include <boost/range/adaptor/reversed.hpp>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
@ -32,8 +32,6 @@
|
|||||||
#include <set>
|
#include <set>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -120,4 +118,3 @@ protected:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -15,14 +15,14 @@
|
|||||||
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
#include <libyul/optimiser/BlockFlattener.h>
|
#include <libyul/optimiser/BlockFlattener.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libdevcore/Visitor.h>
|
#include <libdevcore/Visitor.h>
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void BlockFlattener::operator()(Block& _block)
|
void BlockFlattener::operator()(Block& _block)
|
||||||
{
|
{
|
||||||
|
@ -18,8 +18,6 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -31,4 +29,3 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -24,12 +24,11 @@
|
|||||||
#include <libyul/optimiser/Metrics.h>
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/optimiser/SyntacticalEquality.h>
|
#include <libyul/optimiser/SyntacticalEquality.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void CommonSubexpressionEliminator::visit(Expression& _e)
|
void CommonSubexpressionEliminator::visit(Expression& _e)
|
||||||
{
|
{
|
||||||
|
@ -23,8 +23,6 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/DataFlowAnalyzer.h>
|
#include <libyul/optimiser/DataFlowAnalyzer.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -42,4 +40,3 @@ protected:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -25,8 +25,7 @@
|
|||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -34,7 +33,7 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void DataFlowAnalyzer::operator()(Assignment& _assignment)
|
void DataFlowAnalyzer::operator()(Assignment& _assignment)
|
||||||
{
|
{
|
||||||
|
@ -23,14 +23,11 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -88,4 +85,3 @@ protected:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,17 +21,14 @@
|
|||||||
#include <libyul/optimiser/Disambiguator.h>
|
#include <libyul/optimiser/Disambiguator.h>
|
||||||
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmScope.h>
|
||||||
#include <libsolidity/inlineasm/AsmScope.h>
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
using Scope = dev::solidity::assembly::Scope;
|
|
||||||
|
|
||||||
YulString Disambiguator::translateIdentifier(YulString _originalName)
|
YulString Disambiguator::translateIdentifier(YulString _originalName)
|
||||||
{
|
{
|
||||||
if ((m_externallyUsedIdentifiers.count(_originalName)))
|
if ((m_externallyUsedIdentifiers.count(_originalName)))
|
||||||
|
@ -20,20 +20,16 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libyul/optimiser/ASTCopier.h>
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
#include <libyul/optimiser/NameDispenser.h>
|
#include <libyul/optimiser/NameDispenser.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -44,7 +40,7 @@ class Disambiguator: public ASTCopier
|
|||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
explicit Disambiguator(
|
explicit Disambiguator(
|
||||||
solidity::assembly::AsmAnalysisInfo const& _analysisInfo,
|
AsmAnalysisInfo const& _analysisInfo,
|
||||||
std::set<YulString> const& _externallyUsedIdentifiers = {}
|
std::set<YulString> const& _externallyUsedIdentifiers = {}
|
||||||
):
|
):
|
||||||
m_info(_analysisInfo), m_externallyUsedIdentifiers(_externallyUsedIdentifiers), m_nameDispenser(m_externallyUsedIdentifiers)
|
m_info(_analysisInfo), m_externallyUsedIdentifiers(_externallyUsedIdentifiers), m_nameDispenser(m_externallyUsedIdentifiers)
|
||||||
@ -58,16 +54,15 @@ protected:
|
|||||||
void leaveFunction(FunctionDefinition const& _function) override;
|
void leaveFunction(FunctionDefinition const& _function) override;
|
||||||
YulString translateIdentifier(YulString _name) override;
|
YulString translateIdentifier(YulString _name) override;
|
||||||
|
|
||||||
void enterScopeInternal(solidity::assembly::Scope& _scope);
|
void enterScopeInternal(Scope& _scope);
|
||||||
void leaveScopeInternal(solidity::assembly::Scope& _scope);
|
void leaveScopeInternal(Scope& _scope);
|
||||||
|
|
||||||
solidity::assembly::AsmAnalysisInfo const& m_info;
|
AsmAnalysisInfo const& m_info;
|
||||||
std::set<YulString> const& m_externallyUsedIdentifiers;
|
std::set<YulString> const& m_externallyUsedIdentifiers;
|
||||||
|
|
||||||
std::vector<solidity::assembly::Scope*> m_scopes;
|
std::vector<Scope*> m_scopes;
|
||||||
std::map<void const*, YulString> m_translations;
|
std::map<void const*, YulString> m_translations;
|
||||||
NameDispenser m_nameDispenser;
|
NameDispenser m_nameDispenser;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -23,14 +23,13 @@
|
|||||||
#include <libyul/optimiser/InlinableExpressionFunctionFinder.h>
|
#include <libyul/optimiser/InlinableExpressionFunctionFinder.h>
|
||||||
#include <libyul/optimiser/Substitution.h>
|
#include <libyul/optimiser/Substitution.h>
|
||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <boost/algorithm/cxx11/all_of.hpp>
|
#include <boost/algorithm/cxx11/all_of.hpp>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void ExpressionInliner::run()
|
void ExpressionInliner::run()
|
||||||
|
@ -20,16 +20,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
#include <libyul/AsmDataForward.h>
|
||||||
#include <libyul/ASTDataForward.h>
|
|
||||||
|
|
||||||
#include <boost/variant.hpp>
|
#include <boost/variant.hpp>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -69,4 +66,3 @@ private:
|
|||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -24,8 +24,7 @@
|
|||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
#include <libyul/optimiser/Utilities.h>
|
#include <libyul/optimiser/Utilities.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -33,7 +32,7 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void ExpressionJoiner::operator()(FunctionalInstruction& _instruction)
|
void ExpressionJoiner::operator()(FunctionalInstruction& _instruction)
|
||||||
|
@ -20,14 +20,11 @@
|
|||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -99,4 +96,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -23,14 +23,13 @@
|
|||||||
#include <libyul/optimiser/SimplificationRules.h>
|
#include <libyul/optimiser/SimplificationRules.h>
|
||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
#include <libyul/optimiser/SSAValueTracker.h>
|
#include <libyul/optimiser/SSAValueTracker.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
|
|
||||||
|
@ -20,12 +20,10 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -52,4 +50,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -32,7 +32,7 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void ExpressionSplitter::operator()(FunctionalInstruction& _instruction)
|
void ExpressionSplitter::operator()(FunctionalInstruction& _instruction)
|
||||||
|
@ -20,15 +20,13 @@
|
|||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
#include <libyul/optimiser/NameDispenser.h>
|
#include <libyul/optimiser/NameDispenser.h>
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -83,4 +81,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -15,13 +15,13 @@
|
|||||||
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
#include <libyul/optimiser/ForLoopInitRewriter.h>
|
#include <libyul/optimiser/ForLoopInitRewriter.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void ForLoopInitRewriter::operator()(Block& _block)
|
void ForLoopInitRewriter::operator()(Block& _block)
|
||||||
{
|
{
|
||||||
|
@ -18,8 +18,6 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -36,4 +34,3 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -27,8 +27,7 @@
|
|||||||
#include <libyul/optimiser/Metrics.h>
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/optimiser/SSAValueTracker.h>
|
#include <libyul/optimiser/SSAValueTracker.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
#include <libdevcore/Visitor.h>
|
#include <libdevcore/Visitor.h>
|
||||||
@ -37,7 +36,7 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
FullInliner::FullInliner(Block& _ast, NameDispenser& _dispenser):
|
FullInliner::FullInliner(Block& _ast, NameDispenser& _dispenser):
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTCopier.h>
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
@ -33,8 +33,6 @@
|
|||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -153,4 +151,3 @@ public:
|
|||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,13 +21,13 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/FunctionGrouper.h>
|
#include <libyul/optimiser/FunctionGrouper.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <boost/range/algorithm_ext/erase.hpp>
|
#include <boost/range/algorithm_ext/erase.hpp>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
|
|
||||||
|
@ -21,10 +21,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -43,4 +41,3 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -22,14 +22,13 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/FunctionHoister.h>
|
#include <libyul/optimiser/FunctionHoister.h>
|
||||||
#include <libyul/optimiser/Utilities.h>
|
#include <libyul/optimiser/Utilities.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void FunctionHoister::operator()(Block& _block)
|
void FunctionHoister::operator()(Block& _block)
|
||||||
|
@ -21,12 +21,9 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -49,4 +46,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,12 +21,11 @@
|
|||||||
#include <libyul/optimiser/InlinableExpressionFunctionFinder.h>
|
#include <libyul/optimiser/InlinableExpressionFunctionFinder.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/Utilities.h>
|
#include <libyul/optimiser/Utilities.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void InlinableExpressionFunctionFinder::operator()(Identifier const& _identifier)
|
void InlinableExpressionFunctionFinder::operator()(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
|
@ -20,13 +20,11 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -66,4 +64,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -24,13 +24,13 @@
|
|||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void MainFunction::operator()(Block& _block)
|
void MainFunction::operator()(Block& _block)
|
||||||
|
@ -21,10 +21,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -38,4 +36,3 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,10 +20,10 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/Metrics.h>
|
#include <libyul/optimiser/Metrics.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
size_t CodeSize::codeSize(Statement const& _statement)
|
size_t CodeSize::codeSize(Statement const& _statement)
|
||||||
{
|
{
|
||||||
|
@ -22,8 +22,6 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -49,4 +47,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,11 +20,11 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void NameCollector::operator()(VariableDeclaration const& _varDecl)
|
void NameCollector::operator()(VariableDeclaration const& _varDecl)
|
||||||
{
|
{
|
||||||
|
@ -25,8 +25,6 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -83,4 +81,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,12 +21,11 @@
|
|||||||
#include <libyul/optimiser/NameDispenser.h>
|
#include <libyul/optimiser/NameDispenser.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
NameDispenser::NameDispenser(Block const& _ast):
|
NameDispenser::NameDispenser(Block const& _ast):
|
||||||
NameDispenser(NameCollector(_ast).names())
|
NameDispenser(NameCollector(_ast).names())
|
||||||
|
@ -19,14 +19,12 @@
|
|||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -58,4 +56,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -22,8 +22,7 @@
|
|||||||
#include <libyul/optimiser/RedundantAssignEliminator.h>
|
#include <libyul/optimiser/RedundantAssignEliminator.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -31,7 +30,7 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void RedundantAssignEliminator::operator()(Identifier const& _identifier)
|
void RedundantAssignEliminator::operator()(Identifier const& _identifier)
|
||||||
|
@ -21,14 +21,11 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -190,4 +187,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -23,12 +23,11 @@
|
|||||||
#include <libyul/optimiser/Metrics.h>
|
#include <libyul/optimiser/Metrics.h>
|
||||||
#include <libyul/optimiser/ASTCopier.h>
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void Rematerialiser::visit(Expression& _e)
|
void Rematerialiser::visit(Expression& _e)
|
||||||
{
|
{
|
||||||
|
@ -22,8 +22,6 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/DataFlowAnalyzer.h>
|
#include <libyul/optimiser/DataFlowAnalyzer.h>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -41,4 +39,3 @@ protected:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -23,8 +23,7 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/NameCollector.h>
|
#include <libyul/optimiser/NameCollector.h>
|
||||||
#include <libyul/optimiser/NameDispenser.h>
|
#include <libyul/optimiser/NameDispenser.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -32,7 +31,7 @@
|
|||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
using namespace dev::solidity;
|
using namespace dev::solidity;
|
||||||
|
|
||||||
void SSATransform::operator()(Identifier& _identifier)
|
void SSATransform::operator()(Identifier& _identifier)
|
||||||
|
@ -20,14 +20,11 @@
|
|||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTWalker.h>
|
#include <libyul/optimiser/ASTWalker.h>
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -95,4 +92,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,11 +21,11 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/SSAValueTracker.h>
|
#include <libyul/optimiser/SSAValueTracker.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void SSAValueTracker::operator()(Assignment const& _assignment)
|
void SSAValueTracker::operator()(Assignment const& _assignment)
|
||||||
{
|
{
|
||||||
|
@ -26,8 +26,6 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -54,4 +52,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,8 +21,7 @@
|
|||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libevmasm/SemanticInformation.h>
|
#include <libevmasm/SemanticInformation.h>
|
||||||
|
|
||||||
@ -30,7 +29,7 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
MovableChecker::MovableChecker(Expression const& _expression)
|
MovableChecker::MovableChecker(Expression const& _expression)
|
||||||
{
|
{
|
||||||
|
@ -24,8 +24,6 @@
|
|||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -57,4 +55,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -24,15 +24,14 @@
|
|||||||
#include <libyul/optimiser/ASTCopier.h>
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
#include <libyul/optimiser/SyntacticalEquality.h>
|
#include <libyul/optimiser/SyntacticalEquality.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libevmasm/RuleList.h>
|
#include <libevmasm/RuleList.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace langutil;
|
using namespace langutil;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
|
|
||||||
SimplificationRule<Pattern> const* SimplificationRules::findFirstMatch(
|
SimplificationRule<Pattern> const* SimplificationRules::findFirstMatch(
|
||||||
@ -124,7 +123,7 @@ bool Pattern::matches(Expression const& _expr, map<YulString, Expression const*>
|
|||||||
if (expr->type() != typeid(Literal))
|
if (expr->type() != typeid(Literal))
|
||||||
return false;
|
return false;
|
||||||
Literal const& literal = boost::get<Literal>(*expr);
|
Literal const& literal = boost::get<Literal>(*expr);
|
||||||
if (literal.kind != assembly::LiteralKind::Number)
|
if (literal.kind != LiteralKind::Number)
|
||||||
return false;
|
return false;
|
||||||
if (m_data && *m_data != u256(literal.value.str()))
|
if (m_data && *m_data != u256(literal.value.str()))
|
||||||
return false;
|
return false;
|
||||||
@ -194,7 +193,7 @@ Expression Pattern::toExpression(SourceLocation const& _location) const
|
|||||||
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, assembly::LiteralKind::Number, YulString{formatNumber(*m_data)}, {}};
|
return Literal{_location, LiteralKind::Number, YulString{formatNumber(*m_data)}, {}};
|
||||||
}
|
}
|
||||||
else if (m_kind == PatternKind::Operation)
|
else if (m_kind == PatternKind::Operation)
|
||||||
{
|
{
|
||||||
@ -209,7 +208,7 @@ Expression Pattern::toExpression(SourceLocation const& _location) const
|
|||||||
u256 Pattern::d() const
|
u256 Pattern::d() const
|
||||||
{
|
{
|
||||||
Literal const& literal = boost::get<Literal>(matchGroupValue());
|
Literal const& literal = boost::get<Literal>(matchGroupValue());
|
||||||
assertThrow(literal.kind == assembly::LiteralKind::Number, OptimizerException, "");
|
assertThrow(literal.kind == LiteralKind::Number, OptimizerException, "");
|
||||||
assertThrow(isValidDecimal(literal.value.str()) || isValidHex(literal.value.str()), OptimizerException, "");
|
assertThrow(isValidDecimal(literal.value.str()) || isValidHex(literal.value.str()), OptimizerException, "");
|
||||||
return u256(literal.value.str());
|
return u256(literal.value.str());
|
||||||
}
|
}
|
||||||
|
@ -23,17 +23,14 @@
|
|||||||
#include <libevmasm/ExpressionClasses.h>
|
#include <libevmasm/ExpressionClasses.h>
|
||||||
#include <libevmasm/SimplificationRule.h>
|
#include <libevmasm/SimplificationRule.h>
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <boost/noncopyable.hpp>
|
#include <boost/noncopyable.hpp>
|
||||||
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -86,11 +83,11 @@ public:
|
|||||||
/// Matches any expression.
|
/// Matches any expression.
|
||||||
Pattern(PatternKind _kind = PatternKind::Any): m_kind(_kind) {}
|
Pattern(PatternKind _kind = PatternKind::Any): m_kind(_kind) {}
|
||||||
// Matches a specific constant value.
|
// Matches a specific constant value.
|
||||||
Pattern(unsigned _value): Pattern(u256(_value)) {}
|
Pattern(unsigned _value): Pattern(dev::u256(_value)) {}
|
||||||
// Matches a specific constant value.
|
// Matches a specific constant value.
|
||||||
Pattern(u256 const& _value): m_kind(PatternKind::Constant), m_data(std::make_shared<u256>(_value)) {}
|
Pattern(dev::u256 const& _value): m_kind(PatternKind::Constant), m_data(std::make_shared<dev::u256>(_value)) {}
|
||||||
// Matches a given instruction with given arguments
|
// Matches a given instruction with given arguments
|
||||||
Pattern(solidity::Instruction _instruction, std::vector<Pattern> const& _arguments = {});
|
Pattern(dev::solidity::Instruction _instruction, std::vector<Pattern> const& _arguments = {});
|
||||||
/// Sets this pattern to be part of the match group with the identifier @a _group.
|
/// Sets this pattern to be part of the match group with the identifier @a _group.
|
||||||
/// Inside one rule, all patterns in the same match group have to match expressions from the
|
/// Inside one rule, all patterns in the same match group have to match expressions from the
|
||||||
/// same expression equivalence class.
|
/// same expression equivalence class.
|
||||||
@ -101,9 +98,9 @@ public:
|
|||||||
std::vector<Pattern> arguments() const { return m_arguments; }
|
std::vector<Pattern> arguments() const { return m_arguments; }
|
||||||
|
|
||||||
/// @returns the data of the matched expression if this pattern is part of a match group.
|
/// @returns the data of the matched expression if this pattern is part of a match group.
|
||||||
u256 d() const;
|
dev::u256 d() const;
|
||||||
|
|
||||||
solidity::Instruction instruction() const;
|
dev::solidity::Instruction instruction() const;
|
||||||
|
|
||||||
/// 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.
|
||||||
@ -113,12 +110,11 @@ private:
|
|||||||
Expression const& matchGroupValue() const;
|
Expression const& matchGroupValue() const;
|
||||||
|
|
||||||
PatternKind m_kind = PatternKind::Any;
|
PatternKind m_kind = PatternKind::Any;
|
||||||
solidity::Instruction m_instruction; ///< Only valid if m_kind is Operation
|
dev::solidity::Instruction m_instruction; ///< Only valid if m_kind is Operation
|
||||||
std::shared_ptr<u256> m_data; ///< Only valid if m_kind is Constant
|
std::shared_ptr<dev::u256> m_data; ///< Only valid if m_kind is Constant
|
||||||
std::vector<Pattern> m_arguments;
|
std::vector<Pattern> m_arguments;
|
||||||
unsigned m_matchGroup = 0;
|
unsigned m_matchGroup = 0;
|
||||||
std::map<unsigned, Expression const*>* m_matchGroups = nullptr;
|
std::map<unsigned, Expression const*>* m_matchGroups = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,11 +20,11 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/Substitution.h>
|
#include <libyul/optimiser/Substitution.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
Expression Substitution::translate(Expression const& _expression)
|
Expression Substitution::translate(Expression const& _expression)
|
||||||
{
|
{
|
||||||
|
@ -21,13 +21,10 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/optimiser/ASTCopier.h>
|
#include <libyul/optimiser/ASTCopier.h>
|
||||||
|
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -47,4 +44,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -35,21 +35,19 @@
|
|||||||
#include <libyul/optimiser/SSATransform.h>
|
#include <libyul/optimiser/SSATransform.h>
|
||||||
#include <libyul/optimiser/RedundantAssignEliminator.h>
|
#include <libyul/optimiser/RedundantAssignEliminator.h>
|
||||||
#include <libyul/optimiser/VarDeclPropagator.h>
|
#include <libyul/optimiser/VarDeclPropagator.h>
|
||||||
|
#include <libyul/AsmAnalysisInfo.h>
|
||||||
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmPrinter.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmPrinter.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void OptimiserSuite::run(
|
void OptimiserSuite::run(
|
||||||
Block& _ast,
|
Block& _ast,
|
||||||
solidity::assembly::AsmAnalysisInfo const& _analysisInfo,
|
AsmAnalysisInfo const& _analysisInfo,
|
||||||
set<YulString> const& _externallyUsedIdentifiers
|
set<YulString> const& _externallyUsedIdentifiers
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
@ -20,23 +20,16 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
#include <libyul/YulString.h>
|
#include <libyul/YulString.h>
|
||||||
|
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace solidity
|
|
||||||
{
|
|
||||||
namespace assembly
|
|
||||||
{
|
|
||||||
struct AsmAnalysisInfo;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
|
struct AsmAnalysisInfo;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Optimiser suite that combines all steps and also provides the settings for the heuristics
|
* Optimiser suite that combines all steps and also provides the settings for the heuristics
|
||||||
*/
|
*/
|
||||||
@ -45,11 +38,10 @@ class OptimiserSuite
|
|||||||
public:
|
public:
|
||||||
static void run(
|
static void run(
|
||||||
Block& _ast,
|
Block& _ast,
|
||||||
solidity::assembly::AsmAnalysisInfo const& _analysisInfo,
|
AsmAnalysisInfo const& _analysisInfo,
|
||||||
|
|
||||||
std::set<YulString> const& _externallyUsedIdentifiers = {}
|
std::set<YulString> const& _externallyUsedIdentifiers = {}
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -21,14 +21,13 @@
|
|||||||
#include <libyul/optimiser/SyntacticalEquality.h>
|
#include <libyul/optimiser/SyntacticalEquality.h>
|
||||||
|
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
bool SyntacticalEqualityChecker::equal(Expression const& _e1, Expression const& _e2)
|
bool SyntacticalEqualityChecker::equal(Expression const& _e1, Expression const& _e2)
|
||||||
{
|
{
|
||||||
|
@ -20,12 +20,10 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libyul/ASTDataForward.h>
|
#include <libyul/AsmDataForward.h>
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -47,4 +45,3 @@ protected:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -24,14 +24,13 @@
|
|||||||
#include <libyul/optimiser/Semantics.h>
|
#include <libyul/optimiser/Semantics.h>
|
||||||
#include <libyul/optimiser/Utilities.h>
|
#include <libyul/optimiser/Utilities.h>
|
||||||
#include <libyul/Exceptions.h>
|
#include <libyul/Exceptions.h>
|
||||||
|
#include <libyul/AsmData.h>
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
|
||||||
|
|
||||||
#include <boost/algorithm/cxx11/none_of.hpp>
|
#include <boost/algorithm/cxx11/none_of.hpp>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
UnusedPruner::UnusedPruner(Block& _ast, set<YulString> const& _externallyUsedFunctions)
|
UnusedPruner::UnusedPruner(Block& _ast, set<YulString> const& _externallyUsedFunctions)
|
||||||
{
|
{
|
||||||
|
@ -26,8 +26,6 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
namespace dev
|
|
||||||
{
|
|
||||||
namespace yul
|
namespace yul
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -62,4 +60,3 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#include <libyul/optimiser/Utilities.h>
|
#include <libyul/optimiser/Utilities.h>
|
||||||
|
|
||||||
#include <libsolidity/inlineasm/AsmData.h>
|
#include <libyul/AsmData.h>
|
||||||
|
|
||||||
#include <libdevcore/CommonData.h>
|
#include <libdevcore/CommonData.h>
|
||||||
|
|
||||||
@ -28,9 +28,9 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace dev;
|
using namespace dev;
|
||||||
using namespace dev::yul;
|
using namespace yul;
|
||||||
|
|
||||||
void dev::yul::removeEmptyBlocks(Block& _block)
|
void yul::removeEmptyBlocks(Block& _block)
|
||||||
{
|
{
|
||||||
auto isEmptyBlock = [](Statement const& _st) -> bool {
|
auto isEmptyBlock = [](Statement const& _st) -> bool {
|
||||||
return _st.type() == typeid(Block) && boost::get<Block>(_st).statements.empty();
|
return _st.type() == typeid(Block) && boost::get<Block>(_st).statements.empty();
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user