Support multiple assembly front and backends.

This commit is contained in:
chriseth 2017-05-23 18:57:06 +02:00
parent f2804c49ed
commit eaa13d42a0
4 changed files with 217 additions and 33 deletions

View 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);
}

View 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;
};
}
}

View File

@ -35,8 +35,8 @@
#include <libsolidity/interface/StandardCompiler.h>
#include <libsolidity/interface/SourceReferenceFormatter.h>
#include <libsolidity/interface/GasEstimator.h>
#include <libsolidity/interface/MultiBackendAssemblyStack.h>
#include <libsolidity/formal/Why3Translator.h>
#include <libsolidity/inlineasm/AsmStack.h>
#include <libevmasm/Instruction.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_strEVM = "evm";
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_strGas = "gas";
static string const g_strHelp = "help";
@ -166,7 +166,7 @@ static set<string> const g_machineArgs
{
g_strEVM,
g_strEVM15,
g_streWASM
g_streWasm
};
static void version()
@ -717,22 +717,26 @@ bool CommandLineInterface::processInput()
{
// switch to assembly mode
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))
{
string machine = m_args[g_argMachine].as<string>();
if (machine == g_strEVM)
m_assemblyMachine = AssemblyMachine::EVM;
targetMachine = Machine::EVM;
else if (machine == g_strEVM15)
m_assemblyMachine = AssemblyMachine::EVM15;
else if (machine == g_streWASM)
m_assemblyMachine = AssemblyMachine::eWasm;
targetMachine = Machine::EVM15;
else if (machine == g_streWasm)
targetMachine = Machine::eWasm;
else
{
cerr << "Invalid option for --machine: " << machine << endl;
return false;
}
}
return assemble();
return assemble(input, targetMachine);
}
if (m_args.count(g_argLink))
{
@ -1018,22 +1022,20 @@ void CommandLineInterface::writeLinkedFiles()
writeFile(src.first, src.second);
}
bool CommandLineInterface::assemble()
bool CommandLineInterface::assemble(
MultiBackendAssemblyStack::Input _input,
MultiBackendAssemblyStack::Machine _targetMachine
)
{
bool successful = true;
map<string, shared_ptr<Scanner>> scanners;
map<string, assembly::InlineAssemblyStack> assemblyStacks;
map<string, MultiBackendAssemblyStack> assemblyStacks;
for (auto const& src: m_sourceCodes)
{
auto& stack = assemblyStacks[src.first] = MultiBackendAssemblyStack(_input, _targetMachine);
try
{
auto scanner = make_shared<Scanner>(CharStream(src.second), src.first);
scanners[src.first] = scanner;
if (!assemblyStacks[src.first].parse(scanner))
if (!stack.parseAndAnalyze(src.first, src.second))
successful = false;
else
//@TODO we should not just throw away the result here
assemblyStacks[src.first].assemble();
}
catch (Exception const& _exception)
{
@ -1046,16 +1048,17 @@ bool CommandLineInterface::assemble()
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(
cerr,
*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;
}
@ -1064,10 +1067,27 @@ bool CommandLineInterface::assemble()
for (auto const& src: m_sourceCodes)
{
cout << endl << "======= " << src.first << " =======" << endl;
eth::Assembly assembly = assemblyStacks[src.first].assemble();
cout << assembly.assemble().toHex() << endl;
assembly.stream(cout, "", m_sourceCodes);
string machine =
_targetMachine == AssemblyStack::Machine::EVM ? "EVM" :
_targetMachine == AssemblyStack::Machine::EVM15 ? "EVM 1.5" :
"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;

View File

@ -22,6 +22,7 @@
#pragma once
#include <libsolidity/interface/CompilerStack.h>
#include <libsolidity/interface/MultiBackendAssemblyStack.h>
#include <boost/program_options.hpp>
#include <boost/filesystem/path.hpp>
@ -53,9 +54,7 @@ private:
bool link();
void writeLinkedFiles();
/// Parse assembly input.
bool assemble();
void outputAssembly();
bool assemble(MultiBackendAssemblyStack::Input _input, MultiBackendAssemblyStack::Machine _targetMachine);
void outputCompilationResults();
@ -85,11 +84,6 @@ private:
bool m_error = false; ///< If true, some error occurred.
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;