aboutsummaryrefslogblamecommitdiffstats
path: root/test/libsolidity/SolidityExpressionCompiler.cpp
blob: 26b7914fc4588f41ab2747b6ce32a0c31bbc04df (plain) (tree)
1
2
3
4
5
6
7
8
9
  
                                      
 
                                                                        



                                                                            
                                                                   




                                                                         
                                                                         








                                                   






                                                     
                                                
                         

















                                                                                                
                                                               
        





                                                                                                          












                                                                                                                       
                                      



                                              
 
                                            
                                                                                       
                                                     
         
                                                                           
                                                     
                                                             
         

                                                        

 





                                                                                   
 
                                          

           
                                 

                                                                                                        

                                       





                                                                                                                  




                                                                                

                         
                                            
                                                                     
                                                                          
                                                   
 
                                                               
                                                                  
                                                                                                 
                 
                                                                                                                  
                                                                                              
                 
                                                                  
                                                                                                 
                 
                                                            
                                                                                                       
                                                                                    
                 
                                                                  

                                                                                                 
                                                                      
                                                                         
 
                                                                                        
                                                            
                                                                              

                                                                                                                      
                                                                             



                                                                                                                                      
 
                                                                                     

                                                                        


                                                                                                              
                                                                                




                                                                          
                       







                                                 

                                    
                                                              

                 

                                                        
                                                           




                                                                                                        

                                    
                                                        

                 

                                                        
                                                           




                                                                                                        

                                    
                                                                   

                 

                                                        
                                                                                   



                                                                                                                                          
                                                        

                                    
                               
                                       
                                                
                         

                 

                                                        
                                                           






                                                                                                        

                                                 
                         

                 

                                                        
                                                                                    






                                                                                                        
                                     
                         
                                                   
                         

                 

                                                        
                                                                                               






                                                                                                        
                                       
                                                  
                         

                 

                                                        
                                                                                                     


                                                                                                        

                                

                                    
                                                                            

                 

                                                        
                                                                                                               

                                                                                
                                                                                                                       

                                                                       




                                                                                                        

                                    
                                                                                  

                 

                                                        












                                                                                                       

                                                                                                                               

                                                                       


                                                                                                        
                                
 

                                    
                                                                                                 

                 
                                                                                  







                                                                         
                                                                   




                                                                  

                                                                     

                                                                          
                                                                    

                                                                       
                                                                  

                                                                     

                                                                          
                                                                    

                                                                       

                                                                    




                                                                                                        

                                    
                                                            

                 
                                                                                  
 
                                                         
                                                                   




                                                                         




                                                                                                        

                                    
                                                                                                     

                 


                                                                                                      



                                                                         
                                                                     

                                                                              
                                                                   


                                                                         
                                                                         

                                                                    
                                                                   

                                                                                
                                                                         




                                                                         
                                                                                     


                                                                              
                                                                               



                                                                         
                                                                                 

                                                                               
                                                                           
                                                                                  



                                                                     





                                                                                                        




                                                                    


                                                                                                      



                                                                   
                                                                   


                                                                    
                                                                     
                                                                    
                                                                                                        

 

                                             




                                                        

                                                        
                                                                                                 




                                                                                                        




                                                          

                                                        
                                                                                                    






                                                                                                        

                                    
                                                                                                                                          

                 

                                                        
                                                                   


                                                                                                        

                               


                                      
                                             


                         
 
                                                                                             
                                                                             
 
                                                                                                                                    
 

                                                                          


                                                                                                        



                                    
                                                                   



                                                 
                                            



                                                                                                                                                         
                                                            


                                                                                                        




                           
/*
    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 2014
 * Unit tests for the solidity expression compiler.
 */

#include <string>

#include <libsolidity/parsing/Scanner.h>
#include <libsolidity/parsing/Parser.h>
#include <libsolidity/analysis/NameAndTypeResolver.h>
#include <libsolidity/codegen/CompilerContext.h>
#include <libsolidity/codegen/ExpressionCompiler.h>
#include <libsolidity/ast/AST.h>
#include <libsolidity/analysis/TypeChecker.h>
#include <libsolidity/interface/ErrorReporter.h>
#include <test/Options.h>

using namespace std;

namespace dev
{
namespace solidity
{
namespace test
{

namespace
{

/// Helper class that extracts the first expression in an AST.
class FirstExpressionExtractor: private ASTVisitor
{
public:
    FirstExpressionExtractor(ASTNode& _node): m_expression(nullptr) { _node.accept(*this); }
    Expression* expression() const { return m_expression; }
private:
    virtual bool visit(Assignment& _expression) override { return checkExpression(_expression); }
    virtual bool visit(UnaryOperation& _expression) override { return checkExpression(_expression); }
    virtual bool visit(BinaryOperation& _expression) override { return checkExpression(_expression); }
    virtual bool visit(FunctionCall& _expression) override { return checkExpression(_expression); }
    virtual bool visit(MemberAccess& _expression) override { return checkExpression(_expression); }
    virtual bool visit(IndexAccess& _expression) override { return checkExpression(_expression); }
    virtual bool visit(Identifier& _expression) override { return checkExpression(_expression); }
    virtual bool visit(ElementaryTypeNameExpression& _expression) override { return checkExpression(_expression); }
    virtual bool visit(Literal& _expression) override { return checkExpression(_expression); }
    bool checkExpression(Expression& _expression)
    {
        if (m_expression == nullptr)
            m_expression = &_expression;
        return false;
    }
private:
    Expression* m_expression;
};

Declaration const& resolveDeclaration(
    SourceUnit const& _sourceUnit,
    vector<string> const& _namespacedName,
    NameAndTypeResolver const& _resolver
)
{
    ASTNode const* scope = &_sourceUnit;
    // bracers are required, cause msvc couldn't handle this macro in for statement
    for (string const& namePart: _namespacedName)
    {
        auto declarations = _resolver.resolveName(namePart, scope);
        BOOST_REQUIRE(!declarations.empty());
        BOOST_REQUIRE(scope = *declarations.begin());
    }
    BOOST_REQUIRE(scope);
    return dynamic_cast<Declaration const&>(*scope);
}

bytes compileFirstExpression(
    const string& _sourceCode,
    vector<vector<string>> _functions = {},
    vector<vector<string>> _localVariables = {},
    vector<shared_ptr<MagicVariableDeclaration const>> _globalDeclarations = {}
)
{
    ASTPointer<SourceUnit> sourceUnit;
    try
    {
        ErrorList errors;
        ErrorReporter errorReporter(errors);
        sourceUnit = Parser(errorReporter).parse(make_shared<Scanner>(CharStream(_sourceCode)));
        if (!sourceUnit)
            return bytes();
    }
    catch(boost::exception const& _e)
    {
        auto msg = std::string("Parsing source code failed with: \n") + boost::diagnostic_information(_e);
        BOOST_FAIL(msg);
    }

    vector<Declaration const*> declarations;
    declarations.reserve(_globalDeclarations.size() + 1);
    for (ASTPointer<Declaration const> const& variable: _globalDeclarations)
        declarations.push_back(variable.get());

    ErrorList errors;
    ErrorReporter errorReporter(errors);
    map<ASTNode const*, shared_ptr<DeclarationContainer>> scopes;
    NameAndTypeResolver resolver(declarations, scopes, errorReporter);
    resolver.registerDeclarations(*sourceUnit);

    vector<ContractDefinition const*> inheritanceHierarchy;
    for (ASTPointer<ASTNode> const& node: sourceUnit->nodes())
        if (ContractDefinition* contract = dynamic_cast<ContractDefinition*>(node.get()))
        {
            BOOST_REQUIRE_MESSAGE(resolver.resolveNamesAndTypes(*contract), "Resolving names failed");
            inheritanceHierarchy = vector<ContractDefinition const*>(1, contract);
        }
    for (ASTPointer<ASTNode> const& node: sourceUnit->nodes())
        if (ContractDefinition* contract = dynamic_cast<ContractDefinition*>(node.get()))
        {
            ErrorReporter errorReporter(errors);
            TypeChecker typeChecker(dev::test::Options::get().evmVersion(), errorReporter);
            BOOST_REQUIRE(typeChecker.checkTypeRequirements(*contract));
        }
    for (ASTPointer<ASTNode> const& node: sourceUnit->nodes())
        if (ContractDefinition* contract = dynamic_cast<ContractDefinition*>(node.get()))
        {
            FirstExpressionExtractor extractor(*contract);
            BOOST_REQUIRE(extractor.expression() != nullptr);

            CompilerContext context(dev::test::Options::get().evmVersion());
            context.resetVisitedNodes(contract);
            context.setInheritanceHierarchy(inheritanceHierarchy);
            unsigned parametersSize = _localVariables.size(); // assume they are all one slot on the stack
            context.adjustStackOffset(parametersSize);
            for (vector<string> const& variable: _localVariables)
                context.addVariable(
                    dynamic_cast<VariableDeclaration const&>(resolveDeclaration(*sourceUnit, variable, resolver)),
                    parametersSize--
                );

            ExpressionCompiler(context).compile(*extractor.expression());

            for (vector<string> const& function: _functions)
                context << context.functionEntryLabel(dynamic_cast<FunctionDefinition const&>(
                    resolveDeclaration(*sourceUnit, function, resolver)
                ));
            bytes instructions = context.assembledObject().bytecode;
            // debug
            // cout << eth::disassemble(instructions) << endl;
            return instructions;
        }
    BOOST_FAIL("No contract found in source.");
    return bytes();
}

} // end anonymous namespace

BOOST_AUTO_TEST_SUITE(SolidityExpressionCompiler)

BOOST_AUTO_TEST_CASE(literal_true)
{
    char const* sourceCode = R"(
        contract test {
            function f() public { bool x = true; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH1), 0x1});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(literal_false)
{
    char const* sourceCode = R"(
        contract test {
            function f() { bool x = false; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH1), 0x0});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(int_literal)
{
    char const* sourceCode = R"(
        contract test {
          function f() { uint x = 0x12345678901234567890; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH10), 0x12, 0x34, 0x56, 0x78, 0x90,
                                                       0x12, 0x34, 0x56, 0x78, 0x90});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(int_with_wei_ether_subdenomination)
{
    char const* sourceCode = R"(
        contract test {
            constructor() {
                 uint x = 1 wei;
            }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH1), 0x1});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(int_with_szabo_ether_subdenomination)
{
    char const* sourceCode = R"(
        contract test {
            function test () {
                uint x = 1 szabo;
            }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH5), 0xe8, 0xd4, 0xa5, 0x10, 0x00});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(int_with_finney_ether_subdenomination)
{
    char const* sourceCode = R"(
        contract test {
            constructor()
            {
                 uint x = 1 finney;
            }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH7), 0x3, 0x8d, 0x7e, 0xa4, 0xc6, 0x80, 0x00});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(int_with_ether_ether_subdenomination)
{
    char const* sourceCode = R"(
        contract test {
            constructor() {
                 uint x = 1 ether;
            }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH8), 0xd, 0xe0, 0xb6, 0xb3, 0xa7, 0x64, 0x00, 0x00});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(comparison)
{
    char const* sourceCode = R"(
        contract test {
            function f() { bool x = (0x10aa < 0x11aa) != true; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH1), 0x1, byte(Instruction::ISZERO), byte(Instruction::ISZERO),
                       byte(Instruction::PUSH2), 0x11, 0xaa,
                       byte(Instruction::PUSH2), 0x10, 0xaa,
                       byte(Instruction::LT), byte(Instruction::ISZERO), byte(Instruction::ISZERO),
                       byte(Instruction::EQ),
                       byte(Instruction::ISZERO)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(short_circuiting)
{
    char const* sourceCode = R"(
        contract test {
            function f() { bool x = true != (4 <= 8 + 10 || 9 != 2); }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation({byte(Instruction::PUSH1), 0x12, // 8 + 10
                       byte(Instruction::PUSH1), 0x4,
                       byte(Instruction::GT),
                       byte(Instruction::ISZERO), // after this we have 4 <= 8 + 10
                       byte(Instruction::DUP1),
                       byte(Instruction::PUSH1), 0x11,
                       byte(Instruction::JUMPI), // short-circuit if it is true
                       byte(Instruction::POP),
                       byte(Instruction::PUSH1), 0x2,
                       byte(Instruction::PUSH1), 0x9,
                       byte(Instruction::EQ),
                       byte(Instruction::ISZERO), // after this we have 9 != 2
                       byte(Instruction::JUMPDEST),
                       byte(Instruction::ISZERO), byte(Instruction::ISZERO),
                       byte(Instruction::PUSH1), 0x1, byte(Instruction::ISZERO), byte(Instruction::ISZERO),
                       byte(Instruction::EQ),
                       byte(Instruction::ISZERO)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(arithmetic)
{
    char const* sourceCode = R"(
        contract test {
            function f(uint y) { ((((((((y ^ 8) & 7) | 6) - 5) + 4) % 3) / 2) * 1); }
        }
    )";
    bytes code = compileFirstExpression(sourceCode, {}, {{"test", "f", "y"}});
    bytes expectation({byte(Instruction::PUSH1), 0x1,
                       byte(Instruction::PUSH1), 0x2,
                       byte(Instruction::PUSH1), 0x3,
                       byte(Instruction::PUSH1), 0x4,
                       byte(Instruction::PUSH1), 0x5,
                       byte(Instruction::PUSH1), 0x6,
                       byte(Instruction::PUSH1), 0x7,
                       byte(Instruction::PUSH1), 0x8,
                       byte(Instruction::DUP9),
                       byte(Instruction::XOR),
                       byte(Instruction::AND),
                       byte(Instruction::OR),
                       byte(Instruction::SUB),
                       byte(Instruction::ADD),
                       byte(Instruction::DUP2),
                       byte(Instruction::ISZERO),
                       byte(Instruction::ISZERO),
                       byte(Instruction::PUSH1), 0x1d,
                       byte(Instruction::JUMPI),
                       byte(Instruction::INVALID),
                       byte(Instruction::JUMPDEST),
                       byte(Instruction::MOD),
                       byte(Instruction::DUP2),
                       byte(Instruction::ISZERO),
                       byte(Instruction::ISZERO),
                       byte(Instruction::PUSH1), 0x26,
                       byte(Instruction::JUMPI),
                       byte(Instruction::INVALID),
                       byte(Instruction::JUMPDEST),
                       byte(Instruction::DIV),
                       byte(Instruction::MUL)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(unary_operators)
{
    char const* sourceCode = R"(
        contract test {
            function f(int y) { !(~+- y == 2); }
        }
    )";
    bytes code = compileFirstExpression(sourceCode, {}, {{"test", "f", "y"}});

    bytes expectation({byte(Instruction::PUSH1), 0x2,
                       byte(Instruction::DUP2),
                       byte(Instruction::PUSH1), 0x0,
                       byte(Instruction::SUB),
                       byte(Instruction::NOT),
                       byte(Instruction::EQ),
                       byte(Instruction::ISZERO)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(unary_inc_dec)
{
    char const* sourceCode = R"(
        contract test {
            function f(uint a) public returns (uint x) { x = --a ^ (a-- ^ (++a ^ a++)); }
        }
    )";
    bytes code = compileFirstExpression(sourceCode, {}, {{"test", "f", "a"}, {"test", "f", "x"}});

    // Stack: a, x
    bytes expectation({byte(Instruction::DUP2),
                       byte(Instruction::DUP1),
                       byte(Instruction::PUSH1), 0x1,
                       byte(Instruction::ADD),
                       // Stack here: a x a (a+1)
                       byte(Instruction::SWAP3),
                       byte(Instruction::POP), // first ++
                       // Stack here: (a+1) x a
                       byte(Instruction::DUP3),
                       byte(Instruction::PUSH1), 0x1,
                       byte(Instruction::ADD),
                       // Stack here: (a+1) x a (a+2)
                       byte(Instruction::SWAP3),
                       byte(Instruction::POP),
                       // Stack here: (a+2) x a
                       byte(Instruction::DUP3), // second ++
                       byte(Instruction::XOR),
                       // Stack here: (a+2) x a^(a+2)
                       byte(Instruction::DUP3),
                       byte(Instruction::DUP1),
                       byte(Instruction::PUSH1), 0x1,
                       byte(Instruction::SWAP1),
                       byte(Instruction::SUB),
                       // Stack here: (a+2) x a^(a+2) (a+2) (a+1)
                       byte(Instruction::SWAP4),
                       byte(Instruction::POP), // first --
                       byte(Instruction::XOR),
                       // Stack here: (a+1) x a^(a+2)^(a+2)
                       byte(Instruction::DUP3),
                       byte(Instruction::PUSH1), 0x1,
                       byte(Instruction::SWAP1),
                       byte(Instruction::SUB),
                       // Stack here: (a+1) x a^(a+2)^(a+2) a
                       byte(Instruction::SWAP3),
                       byte(Instruction::POP), // second ++
                       // Stack here: a x a^(a+2)^(a+2)
                       byte(Instruction::DUP3), // will change
                       byte(Instruction::XOR),
                       byte(Instruction::SWAP1),
                       byte(Instruction::POP),
                       byte(Instruction::DUP1)});
                       // Stack here: a x a^(a+2)^(a+2)^a
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(assignment)
{
    char const* sourceCode = R"(
        contract test {
            function f(uint a, uint b) { (a += b) * 2; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode, {}, {{"test", "f", "a"}, {"test", "f", "b"}});

    // Stack: a, b
    bytes expectation({byte(Instruction::PUSH1), 0x2,
                       byte(Instruction::DUP2),
                       byte(Instruction::DUP4),
                       byte(Instruction::ADD),
                       // Stack here: a b 2 a+b
                       byte(Instruction::SWAP3),
                       byte(Instruction::POP),
                       byte(Instruction::DUP3),
                       // Stack here: a+b b 2 a+b
                       byte(Instruction::MUL)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(negative_literals_8bits)
{
    char const* sourceCode = R"(
        contract test {
            function f() { int8 x = -0x80; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation(bytes({byte(Instruction::PUSH32)}) + bytes(31, 0xff) + bytes(1, 0x80));
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(negative_literals_16bits)
{
    char const* sourceCode = R"(
        contract test {
            function f() { int64 x = ~0xabc; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation(bytes({byte(Instruction::PUSH32)}) + bytes(30, 0xff) + bytes{0xf5, 0x43});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(intermediately_overflowing_literals)
{
    // first literal itself is too large for 256 bits but it fits after all constant operations
    // have been applied
    char const* sourceCode = R"(
        contract test {
            function f() { uint8 x = (0x00ffffffffffffffffffffffffffffffffffffffff * 0xffffffffffffffffffffffffff01) & 0xbf; }
        }
    )";
    bytes code = compileFirstExpression(sourceCode);

    bytes expectation(bytes({byte(Instruction::PUSH1), 0xbf}));
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(blockhash)
{
    char const* sourceCode = R"(
        contract test {
            function f() {
                blockhash(3);
            }
        }
    )";

    auto blockhashFun = make_shared<FunctionType>(strings{"uint256"}, strings{"bytes32"},
        FunctionType::Kind::BlockHash, false, StateMutability::View);

    bytes code = compileFirstExpression(sourceCode, {}, {}, {make_shared<MagicVariableDeclaration>("blockhash", blockhashFun)});

    bytes expectation({byte(Instruction::PUSH1), 0x03,
                       byte(Instruction::BLOCKHASH)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_CASE(gas_left)
{
    char const* sourceCode = R"(
        contract test {
            function f() public returns (uint256 val) {
                return gasleft();
            }
        }
    )";
    bytes code = compileFirstExpression(
        sourceCode, {}, {},
        {make_shared<MagicVariableDeclaration>("gasleft", make_shared<FunctionType>(strings(), strings{"uint256"}, FunctionType::Kind::GasLeft))}
    );

    bytes expectation = bytes({byte(Instruction::GAS)});
    BOOST_CHECK_EQUAL_COLLECTIONS(code.begin(), code.end(), expectation.begin(), expectation.end());
}

BOOST_AUTO_TEST_SUITE_END()

}
}
} // end namespaces