|
@ -30,8 +30,10 @@ |
|
|
|
|
|
|
|
|
using namespace std; |
|
|
using namespace std; |
|
|
|
|
|
|
|
|
namespace dev { |
|
|
namespace dev |
|
|
namespace solidity { |
|
|
{ |
|
|
|
|
|
namespace solidity |
|
|
|
|
|
{ |
|
|
|
|
|
|
|
|
void ExpressionCompiler::compileExpression(CompilerContext& _context, Expression const& _expression, bool _optimize) |
|
|
void ExpressionCompiler::compileExpression(CompilerContext& _context, Expression const& _expression, bool _optimize) |
|
|
{ |
|
|
{ |
|
@ -51,8 +53,7 @@ bool ExpressionCompiler::visit(Assignment const& _assignment) |
|
|
_assignment.getRightHandSide().accept(*this); |
|
|
_assignment.getRightHandSide().accept(*this); |
|
|
appendTypeConversion(*_assignment.getRightHandSide().getType(), *_assignment.getType()); |
|
|
appendTypeConversion(*_assignment.getRightHandSide().getType(), *_assignment.getType()); |
|
|
_assignment.getLeftHandSide().accept(*this); |
|
|
_assignment.getLeftHandSide().accept(*this); |
|
|
if (asserts(m_currentLValue.isValid())) |
|
|
solAssert(m_currentLValue.isValid(), "LValue not retrieved."); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("LValue not retrieved.")); |
|
|
|
|
|
|
|
|
|
|
|
Token::Value op = _assignment.getAssignmentOperator(); |
|
|
Token::Value op = _assignment.getAssignmentOperator(); |
|
|
if (op != Token::ASSIGN) // compound assignment
|
|
|
if (op != Token::ASSIGN) // compound assignment
|
|
@ -84,8 +85,7 @@ void ExpressionCompiler::endVisit(UnaryOperation const& _unaryOperation) |
|
|
break; |
|
|
break; |
|
|
case Token::DELETE: // delete
|
|
|
case Token::DELETE: // delete
|
|
|
// @todo semantics change for complex types
|
|
|
// @todo semantics change for complex types
|
|
|
if (asserts(m_currentLValue.isValid())) |
|
|
solAssert(m_currentLValue.isValid(), "LValue not retrieved."); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("LValue not retrieved.")); |
|
|
|
|
|
|
|
|
|
|
|
m_context << u256(0); |
|
|
m_context << u256(0); |
|
|
if (m_currentLValue.storesReferenceOnStack()) |
|
|
if (m_currentLValue.storesReferenceOnStack()) |
|
@ -95,8 +95,7 @@ void ExpressionCompiler::endVisit(UnaryOperation const& _unaryOperation) |
|
|
break; |
|
|
break; |
|
|
case Token::INC: // ++ (pre- or postfix)
|
|
|
case Token::INC: // ++ (pre- or postfix)
|
|
|
case Token::DEC: // -- (pre- or postfix)
|
|
|
case Token::DEC: // -- (pre- or postfix)
|
|
|
if (asserts(m_currentLValue.isValid())) |
|
|
solAssert(m_currentLValue.isValid(), "LValue not retrieved."); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("LValue not retrieved.")); |
|
|
|
|
|
m_currentLValue.retrieveValue(_unaryOperation); |
|
|
m_currentLValue.retrieveValue(_unaryOperation); |
|
|
if (!_unaryOperation.isPrefixOperation()) |
|
|
if (!_unaryOperation.isPrefixOperation()) |
|
|
{ |
|
|
{ |
|
@ -179,8 +178,7 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall) |
|
|
if (_functionCall.isTypeConversion()) |
|
|
if (_functionCall.isTypeConversion()) |
|
|
{ |
|
|
{ |
|
|
//@todo struct construction
|
|
|
//@todo struct construction
|
|
|
if (asserts(_functionCall.getArguments().size() == 1)) |
|
|
solAssert(_functionCall.getArguments().size() == 1, ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
Expression const& firstArgument = *_functionCall.getArguments().front(); |
|
|
Expression const& firstArgument = *_functionCall.getArguments().front(); |
|
|
firstArgument.accept(*this); |
|
|
firstArgument.accept(*this); |
|
|
if (firstArgument.getType()->getCategory() == Type::Category::CONTRACT && |
|
|
if (firstArgument.getType()->getCategory() == Type::Category::CONTRACT && |
|
@ -195,8 +193,7 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall) |
|
|
{ |
|
|
{ |
|
|
FunctionType const& function = dynamic_cast<FunctionType const&>(*_functionCall.getExpression().getType()); |
|
|
FunctionType const& function = dynamic_cast<FunctionType const&>(*_functionCall.getExpression().getType()); |
|
|
vector<ASTPointer<Expression const>> arguments = _functionCall.getArguments(); |
|
|
vector<ASTPointer<Expression const>> arguments = _functionCall.getArguments(); |
|
|
if (asserts(arguments.size() == function.getParameterTypes().size())) |
|
|
solAssert(arguments.size() == function.getParameterTypes().size(), ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
|
|
|
|
|
|
switch (function.getLocation()) |
|
|
switch (function.getLocation()) |
|
|
{ |
|
|
{ |
|
@ -282,12 +279,10 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall) |
|
|
bool ExpressionCompiler::visit(NewExpression const& _newExpression) |
|
|
bool ExpressionCompiler::visit(NewExpression const& _newExpression) |
|
|
{ |
|
|
{ |
|
|
ContractType const* type = dynamic_cast<ContractType const*>(_newExpression.getType().get()); |
|
|
ContractType const* type = dynamic_cast<ContractType const*>(_newExpression.getType().get()); |
|
|
if (asserts(type)) |
|
|
solAssert(type, ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
TypePointers const& types = type->getConstructorType()->getParameterTypes(); |
|
|
TypePointers const& types = type->getConstructorType()->getParameterTypes(); |
|
|
vector<ASTPointer<Expression const>> arguments = _newExpression.getArguments(); |
|
|
vector<ASTPointer<Expression const>> arguments = _newExpression.getArguments(); |
|
|
if (asserts(arguments.size() == types.size())) |
|
|
solAssert(arguments.size() == types.size(), ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
|
|
|
|
|
|
// copy the contracts code into memory
|
|
|
// copy the contracts code into memory
|
|
|
bytes const& bytecode = m_context.getCompiledContract(*_newExpression.getContract()); |
|
|
bytes const& bytecode = m_context.getCompiledContract(*_newExpression.getContract()); |
|
@ -439,8 +434,7 @@ void ExpressionCompiler::endVisit(Literal const& _literal) |
|
|
void ExpressionCompiler::appendAndOrOperatorCode(BinaryOperation const& _binaryOperation) |
|
|
void ExpressionCompiler::appendAndOrOperatorCode(BinaryOperation const& _binaryOperation) |
|
|
{ |
|
|
{ |
|
|
Token::Value const op = _binaryOperation.getOperator(); |
|
|
Token::Value const op = _binaryOperation.getOperator(); |
|
|
if (asserts(op == Token::OR || op == Token::AND)) |
|
|
solAssert(op == Token::OR || op == Token::AND, ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
|
|
|
|
|
|
_binaryOperation.getLeftExpression().accept(*this); |
|
|
_binaryOperation.getLeftExpression().accept(*this); |
|
|
m_context << eth::Instruction::DUP1; |
|
|
m_context << eth::Instruction::DUP1; |
|
@ -592,8 +586,7 @@ void ExpressionCompiler::appendExternalFunctionCall(FunctionType const& _functio |
|
|
vector<ASTPointer<Expression const>> const& _arguments, |
|
|
vector<ASTPointer<Expression const>> const& _arguments, |
|
|
FunctionCallOptions const& _options) |
|
|
FunctionCallOptions const& _options) |
|
|
{ |
|
|
{ |
|
|
if (asserts(_arguments.size() == _functionType.getParameterTypes().size())) |
|
|
solAssert(_arguments.size() == _functionType.getParameterTypes().size(), ""); |
|
|
BOOST_THROW_EXCEPTION(InternalCompilerError()); |
|
|
|
|
|
|
|
|
|
|
|
unsigned dataOffset = _options.bare ? 0 : 1; // reserve one byte for the function index
|
|
|
unsigned dataOffset = _options.bare ? 0 : 1; // reserve one byte for the function index
|
|
|
for (unsigned i = 0; i < _arguments.size(); ++i) |
|
|
for (unsigned i = 0; i < _arguments.size(); ++i) |
|
|