mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Support multiple assembly front and backends.
This commit is contained in:
parent
f2804c49ed
commit
eaa13d42a0
85
libsolidity/interface/MultiBackendAssemblyStack.cpp
Normal file
85
libsolidity/interface/MultiBackendAssemblyStack.cpp
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Full assembly stack that can support EVM-assembly and JULIA as input and EVM, EVM1.5 and
|
||||||
|
* eWasm as output.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include <libsolidity/interface/MultiBackendAssemblyStack.h>
|
||||||
|
|
||||||
|
#include <libsolidity/parsing/Scanner.h>
|
||||||
|
#include <libsolidity/inlineasm/AsmPrinter.h>
|
||||||
|
#include <libsolidity/inlineasm/AsmParser.h>
|
||||||
|
#include <libsolidity/inlineasm/AsmAnalysis.h>
|
||||||
|
#include <libsolidity/inlineasm/AsmCodeGen.h>
|
||||||
|
|
||||||
|
#include <libevmasm/Assembly.h>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
using namespace dev;
|
||||||
|
using namespace dev::solidity;
|
||||||
|
|
||||||
|
|
||||||
|
Scanner const& MultiBackendAssemblyStack::scanner() const
|
||||||
|
{
|
||||||
|
solAssert(m_scanner, "");
|
||||||
|
return *m_scanner;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MultiBackendAssemblyStack::parseAndAnalyze(std::string const& _sourceName, std::string const& _source)
|
||||||
|
{
|
||||||
|
m_analysisSuccessful = false;
|
||||||
|
m_scanner = make_shared<Scanner>(CharStream(_source), _sourceName);
|
||||||
|
m_parserResult = assembly::Parser(m_errors, m_input == Input::JULIA).parse(m_scanner);
|
||||||
|
if (!m_errors.empty())
|
||||||
|
return false;
|
||||||
|
solAssert(m_parserResult, "");
|
||||||
|
|
||||||
|
m_analysisInfo = make_shared<assembly::AsmAnalysisInfo>();
|
||||||
|
assembly::AsmAnalyzer analyzer(*m_analysisInfo, m_errors);
|
||||||
|
m_analysisSuccessful = analyzer.analyze(*m_parserResult);
|
||||||
|
return m_analysisSuccessful;
|
||||||
|
}
|
||||||
|
|
||||||
|
eth::LinkerObject MultiBackendAssemblyStack::assemble()
|
||||||
|
{
|
||||||
|
solAssert(m_analysisSuccessful, "");
|
||||||
|
solAssert(m_parserResult, "");
|
||||||
|
solAssert(m_analysisInfo, "");
|
||||||
|
|
||||||
|
switch (m_targetMachine)
|
||||||
|
{
|
||||||
|
case Machine::EVM:
|
||||||
|
{
|
||||||
|
auto assembly = assembly::CodeGenerator(m_errors).assemble(*m_parserResult, *m_analysisInfo);
|
||||||
|
return assembly.assemble();
|
||||||
|
}
|
||||||
|
case Machine::EVM15:
|
||||||
|
solUnimplemented("EVM 1.5 backend is not yet implemented.");
|
||||||
|
case Machine::eWasm:
|
||||||
|
solUnimplemented("eWasm backend is not yet implemented.");
|
||||||
|
}
|
||||||
|
// unreachable
|
||||||
|
return eth::LinkerObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
string MultiBackendAssemblyStack::print()
|
||||||
|
{
|
||||||
|
solAssert(m_parserResult, "");
|
||||||
|
return assembly::AsmPrinter(m_input == Input::JULIA)(*m_parserResult);
|
||||||
|
}
|
85
libsolidity/interface/MultiBackendAssemblyStack.h
Normal file
85
libsolidity/interface/MultiBackendAssemblyStack.h
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
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/>.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Full assembly stack that can support EVM-assembly and JULIA as input and EVM, EVM1.5 and
|
||||||
|
* eWasm as output.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <libsolidity/interface/Exceptions.h>
|
||||||
|
#include <libsolidity/inlineasm/AsmAnalysisInfo.h>
|
||||||
|
#include <libevmasm/LinkerObject.h>
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace dev
|
||||||
|
{
|
||||||
|
namespace solidity
|
||||||
|
{
|
||||||
|
class Scanner;
|
||||||
|
namespace assembly
|
||||||
|
{
|
||||||
|
struct AsmAnalysisInfo;
|
||||||
|
struct Block;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Full assembly stack that can support EVM-assembly and JULIA as input and EVM, EVM1.5 and
|
||||||
|
* eWasm as output.
|
||||||
|
*/
|
||||||
|
class MultiBackendAssemblyStack
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum class Input { JULIA, Assembly };
|
||||||
|
enum class Machine { EVM, EVM15, eWasm };
|
||||||
|
|
||||||
|
MultiBackendAssemblyStack(Input _input = Input::Assembly, Machine _targetMachine = Machine::EVM):
|
||||||
|
m_input(_input),
|
||||||
|
m_targetMachine(_targetMachine)
|
||||||
|
{}
|
||||||
|
|
||||||
|
/// @returns the scanner used during parsing
|
||||||
|
Scanner const& scanner() const;
|
||||||
|
|
||||||
|
/// Runs parsing and analysis steps, returns false if input cannot be assembled.
|
||||||
|
bool parseAndAnalyze(std::string const& _sourceName, std::string const& _source);
|
||||||
|
|
||||||
|
/// Run the assembly step (should only be called after parseAndAnalyze).
|
||||||
|
eth::LinkerObject assemble();
|
||||||
|
|
||||||
|
ErrorList const& errors() const { return m_errors; }
|
||||||
|
|
||||||
|
/// Pretty-print the input after having parsed it.
|
||||||
|
std::string print();
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
Input m_input = Input::Assembly;
|
||||||
|
Machine m_targetMachine = Machine::EVM;
|
||||||
|
|
||||||
|
std::shared_ptr<Scanner> m_scanner;
|
||||||
|
|
||||||
|
bool m_analysisSuccessful = false;
|
||||||
|
std::shared_ptr<assembly::Block> m_parserResult;
|
||||||
|
std::shared_ptr<assembly::AsmAnalysisInfo> m_analysisInfo;
|
||||||
|
ErrorList m_errors;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -35,8 +35,8 @@
|
|||||||
#include <libsolidity/interface/StandardCompiler.h>
|
#include <libsolidity/interface/StandardCompiler.h>
|
||||||
#include <libsolidity/interface/SourceReferenceFormatter.h>
|
#include <libsolidity/interface/SourceReferenceFormatter.h>
|
||||||
#include <libsolidity/interface/GasEstimator.h>
|
#include <libsolidity/interface/GasEstimator.h>
|
||||||
|
#include <libsolidity/interface/MultiBackendAssemblyStack.h>
|
||||||
#include <libsolidity/formal/Why3Translator.h>
|
#include <libsolidity/formal/Why3Translator.h>
|
||||||
#include <libsolidity/inlineasm/AsmStack.h>
|
|
||||||
|
|
||||||
#include <libevmasm/Instruction.h>
|
#include <libevmasm/Instruction.h>
|
||||||
#include <libevmasm/GasMeter.h>
|
#include <libevmasm/GasMeter.h>
|
||||||
@ -83,7 +83,7 @@ static string const g_strCombinedJson = "combined-json";
|
|||||||
static string const g_strContracts = "contracts";
|
static string const g_strContracts = "contracts";
|
||||||
static string const g_strEVM = "evm";
|
static string const g_strEVM = "evm";
|
||||||
static string const g_strEVM15 = "evm15";
|
static string const g_strEVM15 = "evm15";
|
||||||
static string const g_streWASM = "ewasm";
|
static string const g_streWasm = "ewasm";
|
||||||
static string const g_strFormal = "formal";
|
static string const g_strFormal = "formal";
|
||||||
static string const g_strGas = "gas";
|
static string const g_strGas = "gas";
|
||||||
static string const g_strHelp = "help";
|
static string const g_strHelp = "help";
|
||||||
@ -166,7 +166,7 @@ static set<string> const g_machineArgs
|
|||||||
{
|
{
|
||||||
g_strEVM,
|
g_strEVM,
|
||||||
g_strEVM15,
|
g_strEVM15,
|
||||||
g_streWASM
|
g_streWasm
|
||||||
};
|
};
|
||||||
|
|
||||||
static void version()
|
static void version()
|
||||||
@ -717,22 +717,26 @@ bool CommandLineInterface::processInput()
|
|||||||
{
|
{
|
||||||
// switch to assembly mode
|
// switch to assembly mode
|
||||||
m_onlyAssemble = true;
|
m_onlyAssemble = true;
|
||||||
|
using Input = MultiBackendAssemblyStack::Input;
|
||||||
|
using Machine = MultiBackendAssemblyStack::Machine;
|
||||||
|
Input input = m_args.count(g_argJulia) ? Input::JULIA : Input::Assembly;
|
||||||
|
Machine targetMachine = Machine::EVM;
|
||||||
if (m_args.count(g_argMachine))
|
if (m_args.count(g_argMachine))
|
||||||
{
|
{
|
||||||
string machine = m_args[g_argMachine].as<string>();
|
string machine = m_args[g_argMachine].as<string>();
|
||||||
if (machine == g_strEVM)
|
if (machine == g_strEVM)
|
||||||
m_assemblyMachine = AssemblyMachine::EVM;
|
targetMachine = Machine::EVM;
|
||||||
else if (machine == g_strEVM15)
|
else if (machine == g_strEVM15)
|
||||||
m_assemblyMachine = AssemblyMachine::EVM15;
|
targetMachine = Machine::EVM15;
|
||||||
else if (machine == g_streWASM)
|
else if (machine == g_streWasm)
|
||||||
m_assemblyMachine = AssemblyMachine::eWasm;
|
targetMachine = Machine::eWasm;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
cerr << "Invalid option for --machine: " << machine << endl;
|
cerr << "Invalid option for --machine: " << machine << endl;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return assemble();
|
return assemble(input, targetMachine);
|
||||||
}
|
}
|
||||||
if (m_args.count(g_argLink))
|
if (m_args.count(g_argLink))
|
||||||
{
|
{
|
||||||
@ -1018,22 +1022,20 @@ void CommandLineInterface::writeLinkedFiles()
|
|||||||
writeFile(src.first, src.second);
|
writeFile(src.first, src.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CommandLineInterface::assemble()
|
bool CommandLineInterface::assemble(
|
||||||
|
MultiBackendAssemblyStack::Input _input,
|
||||||
|
MultiBackendAssemblyStack::Machine _targetMachine
|
||||||
|
)
|
||||||
{
|
{
|
||||||
bool successful = true;
|
bool successful = true;
|
||||||
map<string, shared_ptr<Scanner>> scanners;
|
map<string, MultiBackendAssemblyStack> assemblyStacks;
|
||||||
map<string, assembly::InlineAssemblyStack> assemblyStacks;
|
|
||||||
for (auto const& src: m_sourceCodes)
|
for (auto const& src: m_sourceCodes)
|
||||||
{
|
{
|
||||||
|
auto& stack = assemblyStacks[src.first] = MultiBackendAssemblyStack(_input, _targetMachine);
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
auto scanner = make_shared<Scanner>(CharStream(src.second), src.first);
|
if (!stack.parseAndAnalyze(src.first, src.second))
|
||||||
scanners[src.first] = scanner;
|
|
||||||
if (!assemblyStacks[src.first].parse(scanner))
|
|
||||||
successful = false;
|
successful = false;
|
||||||
else
|
|
||||||
//@TODO we should not just throw away the result here
|
|
||||||
assemblyStacks[src.first].assemble();
|
|
||||||
}
|
}
|
||||||
catch (Exception const& _exception)
|
catch (Exception const& _exception)
|
||||||
{
|
{
|
||||||
@ -1046,16 +1048,17 @@ bool CommandLineInterface::assemble()
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (auto const& stack: assemblyStacks)
|
for (auto const& sourceAndStack: assemblyStacks)
|
||||||
{
|
{
|
||||||
for (auto const& error: stack.second.errors())
|
auto const& stack = sourceAndStack.second;
|
||||||
|
for (auto const& error: stack.errors())
|
||||||
SourceReferenceFormatter::printExceptionInformation(
|
SourceReferenceFormatter::printExceptionInformation(
|
||||||
cerr,
|
cerr,
|
||||||
*error,
|
*error,
|
||||||
(error->type() == Error::Type::Warning) ? "Warning" : "Error",
|
(error->type() == Error::Type::Warning) ? "Warning" : "Error",
|
||||||
[&](string const& _source) -> Scanner const& { return *scanners.at(_source); }
|
[&](string const&) -> Scanner const& { return stack.scanner(); }
|
||||||
);
|
);
|
||||||
if (!Error::containsOnlyWarnings(stack.second.errors()))
|
if (!Error::containsOnlyWarnings(stack.errors()))
|
||||||
successful = false;
|
successful = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1064,10 +1067,27 @@ bool CommandLineInterface::assemble()
|
|||||||
|
|
||||||
for (auto const& src: m_sourceCodes)
|
for (auto const& src: m_sourceCodes)
|
||||||
{
|
{
|
||||||
cout << endl << "======= " << src.first << " =======" << endl;
|
string machine =
|
||||||
eth::Assembly assembly = assemblyStacks[src.first].assemble();
|
_targetMachine == AssemblyStack::Machine::EVM ? "EVM" :
|
||||||
cout << assembly.assemble().toHex() << endl;
|
_targetMachine == AssemblyStack::Machine::EVM15 ? "EVM 1.5" :
|
||||||
assembly.stream(cout, "", m_sourceCodes);
|
"eWasm";
|
||||||
|
cout << endl << "======= " << src.first << " (" << machine << ") =======" << endl;
|
||||||
|
AssemblyStack& stack = assemblyStacks[src.first];
|
||||||
|
try
|
||||||
|
{
|
||||||
|
cout << stack.assemble(_targetMachine).toHex() << endl;
|
||||||
|
}
|
||||||
|
catch (Exception const& _exception)
|
||||||
|
{
|
||||||
|
cerr << "Exception while assembling: " << boost::diagnostic_information(_exception) << endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
cerr << "Unknown exception while assembling." << endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
cout << stack.print() << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <libsolidity/interface/CompilerStack.h>
|
#include <libsolidity/interface/CompilerStack.h>
|
||||||
|
#include <libsolidity/interface/MultiBackendAssemblyStack.h>
|
||||||
|
|
||||||
#include <boost/program_options.hpp>
|
#include <boost/program_options.hpp>
|
||||||
#include <boost/filesystem/path.hpp>
|
#include <boost/filesystem/path.hpp>
|
||||||
@ -53,9 +54,7 @@ private:
|
|||||||
bool link();
|
bool link();
|
||||||
void writeLinkedFiles();
|
void writeLinkedFiles();
|
||||||
|
|
||||||
/// Parse assembly input.
|
bool assemble(MultiBackendAssemblyStack::Input _input, MultiBackendAssemblyStack::Machine _targetMachine);
|
||||||
bool assemble();
|
|
||||||
void outputAssembly();
|
|
||||||
|
|
||||||
void outputCompilationResults();
|
void outputCompilationResults();
|
||||||
|
|
||||||
@ -85,11 +84,6 @@ private:
|
|||||||
bool m_error = false; ///< If true, some error occurred.
|
bool m_error = false; ///< If true, some error occurred.
|
||||||
|
|
||||||
bool m_onlyAssemble = false;
|
bool m_onlyAssemble = false;
|
||||||
/// Settings to use in assembly / JULIA mode.
|
|
||||||
enum class AssemblyInput { JULIA, Assembly };
|
|
||||||
enum class AssemblyMachine { EVM, EVM15, eWasm };
|
|
||||||
AssemblyInput m_assemblyInput = AssemblyInput::Assembly;
|
|
||||||
AssemblyMachine m_assemblyMachine = AssemblyMachine::EVM;
|
|
||||||
|
|
||||||
bool m_onlyLink = false;
|
bool m_onlyLink = false;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user