Add global variables and support multi-return and multi-assignment.

This commit is contained in:
chriseth 2019-04-23 10:45:16 +02:00
parent 61e36cbfaa
commit c3705f268c
7 changed files with 130 additions and 45 deletions

View File

@ -30,24 +30,32 @@ namespace wasm
{
struct Literal;
struct Identifier;
struct LocalVariable;
struct GlobalVariable;
struct Label;
struct FunctionCall;
struct BuiltinCall;
struct LocalAssignment;
struct GlobalAssignment;
struct Block;
struct If;
struct Loop;
struct Break;
struct Continue;
using Expression = boost::variant<Literal, Identifier, Label, FunctionCall, BuiltinCall, LocalAssignment, Block, If, Loop, Break, Continue>;
using Expression = boost::variant<
Literal, LocalVariable, GlobalVariable, Label,
FunctionCall, BuiltinCall, LocalAssignment, GlobalAssignment,
Block, If, Loop, Break, Continue
>;
struct Literal { uint64_t value; };
struct Identifier { std::string name; };
struct LocalVariable { std::string name; };
struct GlobalVariable { std::string name; };
struct Label { std::string name; };
struct FunctionCall { std::string functionName; std::vector<Expression> arguments; };
struct BuiltinCall { std::string functionName; std::vector<Expression> arguments; };
struct LocalAssignment { std::string variableName; std::unique_ptr<Expression> value; };
struct GlobalAssignment { std::string variableName; std::unique_ptr<Expression> value; };
struct Block { std::string labelName; std::vector<Expression> statements; };
struct If { std::unique_ptr<Expression> condition; std::vector<Expression> statements; };
struct Loop { std::string labelName; std::vector<Expression> statements; };
@ -55,6 +63,8 @@ struct Break { Label label; };
struct Continue { Label label; };
struct VariableDeclaration { std::string variableName; };
struct GlobalVariableDeclaration { std::string variableName; };
struct FunctionDefinition
{
std::string name;

View File

@ -21,6 +21,7 @@
#include <libyul/backends/wasm/EWasmCodeTransform.h>
#include <libyul/backends/wasm/EWasmToText.h>
#include <libyul/optimiser/NameCollector.h>
#include <libyul/AsmData.h>
#include <libyul/Dialect.h>
@ -30,50 +31,68 @@
#include <liblangutil/Exceptions.h>
#include <boost/range/adaptor/reversed.hpp>
#include <boost/range/adaptor/transformed.hpp>
using namespace std;
using namespace dev;
using namespace yul;
string EWasmCodeTransform::run(yul::Block const& _ast)
string EWasmCodeTransform::run(Dialect const& _dialect, yul::Block const& _ast)
{
EWasmCodeTransform transform(_dialect, _ast);
vector<wasm::FunctionDefinition> functions;
for (auto const& statement: _ast.statements)
{
yulAssert(statement.type() == typeid(yul::FunctionDefinition), "");
functions.emplace_back(translateFunction(boost::get<yul::FunctionDefinition>(statement)));
functions.emplace_back(transform.translateFunction(boost::get<yul::FunctionDefinition>(statement)));
}
return EWasmToText{}.run(functions);
return EWasmToText{}.run(transform.m_globalVariables, functions);
}
wasm::Expression EWasmCodeTransform::generateMultiAssignment(
vector<string> _variableNames,
unique_ptr<wasm::Expression> _firstValue
)
{
yulAssert(!_variableNames.empty(), "");
wasm::LocalAssignment assignment{std::move(_variableNames.front()), std::move(_firstValue)};
if (_variableNames.size() == 1)
return move(assignment);
wasm::Block block;
block.statements.emplace_back(std::move(assignment));
for (size_t i = 1; i < _variableNames.size(); ++i)
block.statements.emplace_back(wasm::LocalAssignment{
std::move(_variableNames.at(i)),
make_unique<wasm::Expression>(wasm::GlobalVariable{m_globalVariables.at(i - 1).variableName})
});
return std::move(block);
}
wasm::Expression EWasmCodeTransform::operator()(VariableDeclaration const& _varDecl)
{
vector<string> variableNames;
for (auto const& var: _varDecl.variables)
m_localVariables.emplace_back(wasm::VariableDeclaration{var.name.str()});
{
variableNames.emplace_back(var.name.str());
m_localVariables.emplace_back(wasm::VariableDeclaration{variableNames.back()});
}
if (_varDecl.value)
{
// TODO otherwise, we have to work with globals.
solUnimplementedAssert(_varDecl.variables.size() == 1, "Only single-variable assignments supported.");
return wasm::LocalAssignment{
_varDecl.variables.front().name.str(),
visit(*_varDecl.value)
};
}
return generateMultiAssignment(std::move(variableNames), visit(*_varDecl.value));
else
// TODO this could be handled better.
return wasm::BuiltinCall{"nop", {}};
}
wasm::Expression EWasmCodeTransform::operator()(Assignment const& _assignment)
{
solUnimplementedAssert(_assignment.variableNames.size() == 1, "Only single-variable assignments supported.");
return wasm::LocalAssignment{
_assignment.variableNames.front().name.str(),
visit(*_assignment.value)
};
vector<string> variableNames;
for (auto const& var: _assignment.variableNames)
variableNames.emplace_back(var.name.str());
return generateMultiAssignment(std::move(variableNames), visit(*_assignment.value));
}
wasm::Expression EWasmCodeTransform::operator()(StackAssignment const&)
@ -104,12 +123,16 @@ wasm::Expression EWasmCodeTransform::operator()(FunctionCall const& _call)
if (m_dialect.builtin(_call.functionName.name))
return wasm::BuiltinCall{_call.functionName.name.str(), visit(_call.arguments)};
else
// If this function returns multiple values, then the first one will
// be returned in the expression itself and the others in global variables.
// The values have to be used right away in an assignment or variable declaration,
// so it is handled there.
return wasm::FunctionCall{_call.functionName.name.str(), visit(_call.arguments)};
}
wasm::Expression EWasmCodeTransform::operator()(Identifier const& _identifier)
{
return wasm::Identifier{_identifier.name.str()};
return wasm::LocalVariable{_identifier.name.str()};
}
wasm::Expression EWasmCodeTransform::operator()(Literal const& _literal)
@ -121,7 +144,7 @@ wasm::Expression EWasmCodeTransform::operator()(Literal const& _literal)
wasm::Expression EWasmCodeTransform::operator()(yul::Instruction const&)
{
yulAssert(false, "");
yulAssert(false, "EVM instruction used for Wasm code generation.");
return {};
}
@ -229,14 +252,31 @@ wasm::FunctionDefinition EWasmCodeTransform::translateFunction(yul::FunctionDefi
fun.locals += m_localVariables;
m_localVariables.clear();
yulAssert(_fun.returnVariables.size() <= 1, "");
if (_fun.returnVariables.size() == 1)
fun.body.emplace_back(wasm::Identifier{_fun.returnVariables.front().name.str()});
if (!_fun.returnVariables.empty())
{
// First return variable is returned directly, the others are stored
// in globals.
allocateGlobals(_fun.returnVariables.size() - 1);
for (size_t i = 1; i < _fun.returnVariables.size(); ++i)
fun.body.emplace_back(wasm::GlobalAssignment{
m_globalVariables.at(i - 1).variableName,
make_unique<wasm::Expression>(wasm::LocalVariable{_fun.returnVariables.at(i).name.str()})
});
fun.body.emplace_back(wasm::LocalVariable{_fun.returnVariables.front().name.str()});
}
return fun;
}
string EWasmCodeTransform::newLabel()
{
// TODO this should not clash with other identifiers!
return "label_" + to_string(++m_labelCounter);
return m_nameDispenser.newName("label_"_yulstring).str();
}
void EWasmCodeTransform::allocateGlobals(size_t _amount)
{
while (m_globalVariables.size() < _amount)
m_globalVariables.emplace_back(wasm::GlobalVariableDeclaration{
m_nameDispenser.newName("global_"_yulstring).str()
});
}

View File

@ -23,6 +23,7 @@
#include <libyul/backends/wasm/EWasmAST.h>
#include <libyul/AsmDataForward.h>
#include <libyul/Dialect.h>
#include <libyul/optimiser/NameDispenser.h>
#include <stack>
@ -33,14 +34,7 @@ struct AsmAnalysisInfo;
class EWasmCodeTransform: public boost::static_visitor<wasm::Expression>
{
public:
EWasmCodeTransform(
AsmAnalysisInfo&,
Dialect const& _dialect
):
m_dialect(_dialect)
{}
std::string run(yul::Block const& _ast);
static std::string run(Dialect const& _dialect, yul::Block const& _ast);
public:
wasm::Expression operator()(yul::Instruction const& _instruction);
@ -62,21 +56,41 @@ public:
wasm::Expression operator()(yul::Block const& _block);
private:
EWasmCodeTransform(
Dialect const& _dialect,
Block const& _ast
):
m_dialect(_dialect),
m_nameDispenser(_dialect, _ast)
{}
std::unique_ptr<wasm::Expression> visit(yul::Expression const& _expression);
wasm::Expression visitReturnByValue(yul::Expression const& _expression);
std::vector<wasm::Expression> visit(std::vector<yul::Expression> const& _expressions);
wasm::Expression visit(yul::Statement const& _statement);
std::vector<wasm::Expression> visit(std::vector<yul::Statement> const& _statements);
/// Returns an assignment or a block containing multiple assignments.
/// @param _variableNames the names of the variables to assign to
/// @param _firstValue the value to be assigned to the first variable. If there
/// is more than one variable, the values are taken from m_globalVariables.
wasm::Expression generateMultiAssignment(
std::vector<std::string> _variableNames,
std::unique_ptr<wasm::Expression> _firstValue
);
wasm::FunctionDefinition translateFunction(yul::FunctionDefinition const& _funDef);
std::string newLabel();
std::vector<wasm::VariableDeclaration> m_localVariables;
size_t m_labelCounter = 0;
std::stack<std::pair<std::string, std::string>> m_breakContinueLabelNames;
/// Makes sure that there are at least @a _amount global variables.
void allocateGlobals(size_t _amount);
Dialect const& m_dialect;
NameDispenser m_nameDispenser;
std::vector<wasm::VariableDeclaration> m_localVariables;
std::vector<wasm::GlobalVariableDeclaration> m_globalVariables;
std::stack<std::pair<std::string, std::string>> m_breakContinueLabelNames;
};
}

View File

@ -46,7 +46,7 @@ string EWasmObjectCompiler::run(Object& _object)
yulAssert(_object.analysisInfo, "No analysis info.");
yulAssert(_object.code, "No code.");
ret += EWasmCodeTransform{*_object.analysisInfo, m_dialect}.run(*_object.code);
ret += EWasmCodeTransform::run(m_dialect, *_object.code);
return ret;
}

View File

@ -29,9 +29,14 @@
using namespace std;
using namespace yul;
string EWasmToText::run(vector<wasm::FunctionDefinition> const& _functions)
string EWasmToText::run(
vector<wasm::GlobalVariableDeclaration> const& _globals,
vector<wasm::FunctionDefinition> const& _functions
)
{
string ret = "(module\n\n";
for (auto const& g: _globals)
ret += " (global $" + g.variableName + " (mut i64) (i64.const 0))\n";
for (auto const& f: _functions)
ret += transform(f) + "\n";
return move(ret) + ")\n";
@ -42,11 +47,16 @@ string EWasmToText::operator()(wasm::Literal const& _literal)
return "(i64.const " + to_string(_literal.value) + ")";
}
string EWasmToText::operator()(wasm::Identifier const& _identifier)
string EWasmToText::operator()(wasm::LocalVariable const& _identifier)
{
return "(get_local $" + _identifier.name + ")";
}
string EWasmToText::operator()(wasm::GlobalVariable const& _identifier)
{
return "(get_global $" + _identifier.name + ")";
}
string EWasmToText::operator()(wasm::Label const& _label)
{
return "$" + _label.name;
@ -67,6 +77,11 @@ string EWasmToText::operator()(wasm::LocalAssignment const& _assignment)
return "(set_local $" + _assignment.variableName + " " + visit(*_assignment.value) + ")\n";
}
string EWasmToText::operator()(wasm::GlobalAssignment const& _assignment)
{
return "(set_global $" + _assignment.variableName + " " + visit(*_assignment.value) + ")\n";
}
string EWasmToText::operator()(wasm::If const& _if)
{
return "(if " + visit(*_if.condition) + " (then\n" + indented(joinTransformed(_if.statements)) + "\n))\n";

View File

@ -31,15 +31,20 @@ struct AsmAnalysisInfo;
class EWasmToText: public boost::static_visitor<std::string>
{
public:
std::string run(std::vector<wasm::FunctionDefinition> const& _functions);
std::string run(
std::vector<wasm::GlobalVariableDeclaration> const& _globals,
std::vector<wasm::FunctionDefinition> const& _functions
);
public:
std::string operator()(wasm::Literal const& _literal);
std::string operator()(wasm::Identifier const& _identifier);
std::string operator()(wasm::LocalVariable const& _identifier);
std::string operator()(wasm::GlobalVariable const& _identifier);
std::string operator()(wasm::Label const& _label);
std::string operator()(wasm::BuiltinCall const& _builinCall);
std::string operator()(wasm::FunctionCall const& _functionCall);
std::string operator()(wasm::LocalAssignment const& _assignment);
std::string operator()(wasm::GlobalAssignment const& _assignment);
std::string operator()(wasm::If const& _if);
std::string operator()(wasm::Loop const& _loop);
std::string operator()(wasm::Break const& _break);

View File

@ -2,3 +2,4 @@ iff
nd
assignend
uint
mut