@ -21,6 +21,7 @@
*/
# include <algorithm>
# include <boost/range/adaptor/reversed.hpp>
# include <libevmcore/Instruction.h>
# include <libevmcore/Assembly.h>
# include <libsolidity/AST.h>
@ -34,48 +35,84 @@ using namespace std;
namespace dev {
namespace solidity {
void Compiler : : compileContract ( ContractDefinition const & _contract , vector < MagicVariableDeclaration const * > const & _magicGlobals ,
void Compiler : : compileContract ( ContractDefinition const & _contract ,
map < ContractDefinition const * , bytes const * > const & _contracts )
{
m_context = CompilerContext ( ) ; // clear it just in case
initializeContext ( _contract , _magicGlobals , _ contracts ) ;
initializeContext ( _contract , _contracts ) ;
for ( ASTPointer < FunctionDefinition > const & function : _contract . getDefinedFunctions ( ) )
if ( function - > getName ( ) ! = _contract . getName ( ) ) // don't add the constructor here
for ( ContractDefinition const * contract : _contract . getLinearizedBaseContracts ( ) )
for ( ASTPointer < FunctionDefinition > const & function : contract - > getDefinedFunctions ( ) )
if ( function - > getName ( ) ! = contract - > getName ( ) ) // don't add the constructor here
m_context . addFunction ( * function ) ;
appendFunctionSelector ( _contract ) ;
for ( ASTPointer < FunctionDefinition > const & function : _contract . getDefinedFunctions ( ) )
if ( function - > getName ( ) ! = _contract . getName ( ) ) // don't add the constructor here
for ( ContractDefinition const * contract : _contract . getLinearizedBaseContracts ( ) )
for ( ASTPointer < FunctionDefinition > const & function : contract - > getDefinedFunctions ( ) )
if ( function - > getName ( ) ! = contract - > getName ( ) ) // don't add the constructor here
function - > accept ( * this ) ;
// Swap the runtime context with the creation-time context
swap ( m_context , m_runtimeContext ) ;
initializeContext ( _contract , _magicGlobals , _ contracts ) ;
initializeContext ( _contract , _contracts ) ;
packIntoContractCreator ( _contract , m_runtimeContext ) ;
}
void Compiler : : initializeContext ( ContractDefinition const & _contract , vector < MagicVariableDeclaration const * > const & _magicGlobals ,
void Compiler : : initializeContext ( ContractDefinition const & _contract ,
map < ContractDefinition const * , bytes const * > const & _contracts )
{
m_context . setCompiledContracts ( _contracts ) ;
for ( MagicVariableDeclaration const * variable : _magicGlobals )
m_context . addMagicGlobal ( * variable ) ;
registerStateVariables ( _contract ) ;
}
void Compiler : : packIntoContractCreator ( ContractDefinition const & _contract , CompilerContext const & _runtimeContext )
{
// arguments for base constructors, filled in derived-to-base order
map < ContractDefinition const * , vector < ASTPointer < Expression > > const * > baseArguments ;
set < FunctionDefinition const * > neededFunctions ;
FunctionDefinition const * constructor = _contract . getConstructor ( ) ;
if ( constructor )
neededFunctions = getFunctionsNeededByConstructor ( * constructor ) ;
set < ASTNode const * > nodesUsedInConstructors ;
// Determine the arguments that are used for the base constructors and also which functions
// are needed at compile time.
std : : vector < ContractDefinition const * > const & bases = _contract . getLinearizedBaseContracts ( ) ;
for ( ContractDefinition const * contract : bases )
{
if ( FunctionDefinition const * constructor = contract - > getConstructor ( ) )
nodesUsedInConstructors . insert ( constructor ) ;
for ( ASTPointer < InheritanceSpecifier > const & base : contract - > getBaseContracts ( ) )
{
ContractDefinition const * baseContract = dynamic_cast < ContractDefinition const * > (
base - > getName ( ) - > getReferencedDeclaration ( ) ) ;
solAssert ( baseContract , " " ) ;
if ( baseArguments . count ( baseContract ) = = 0 )
{
baseArguments [ baseContract ] = & base - > getArguments ( ) ;
for ( ASTPointer < Expression > const & arg : base - > getArguments ( ) )
nodesUsedInConstructors . insert ( arg . get ( ) ) ;
}
}
}
//@TODO add virtual functions
neededFunctions = getFunctionsCalled ( nodesUsedInConstructors ) ;
for ( FunctionDefinition const * fun : neededFunctions )
m_context . addFunction ( * fun ) ;
if ( constructor )
appendConstructorCall ( * constructor ) ;
// Call constructors in base-to-derived order.
// The Constructor for the most derived contract is called later.
for ( unsigned i = 1 ; i < bases . size ( ) ; i + + )
{
ContractDefinition const * base = bases [ bases . size ( ) - i ] ;
solAssert ( base , " " ) ;
FunctionDefinition const * baseConstructor = base - > getConstructor ( ) ;
if ( ! baseConstructor )
continue ;
solAssert ( baseArguments [ base ] , " " ) ;
appendBaseConstructorCall ( * baseConstructor , * baseArguments [ base ] ) ;
}
if ( _contract . getConstructor ( ) )
appendConstructorCall ( * _contract . getConstructor ( ) ) ;
eth : : AssemblyItem sub = m_context . addSubroutine ( _runtimeContext . getAssembly ( ) ) ;
// stack contains sub size
@ -88,6 +125,21 @@ void Compiler::packIntoContractCreator(ContractDefinition const& _contract, Comp
fun - > accept ( * this ) ;
}
void Compiler : : appendBaseConstructorCall ( FunctionDefinition const & _constructor ,
vector < ASTPointer < Expression > > const & _arguments )
{
FunctionType constructorType ( _constructor ) ;
eth : : AssemblyItem returnLabel = m_context . pushNewTag ( ) ;
for ( unsigned i = 0 ; i < _arguments . size ( ) ; + + i )
{
compileExpression ( * _arguments [ i ] ) ;
ExpressionCompiler : : appendTypeConversion ( m_context , * _arguments [ i ] - > getType ( ) ,
* constructorType . getParameterTypes ( ) [ i ] ) ;
}
m_context . appendJumpTo ( m_context . getFunctionEntryLabel ( _constructor ) ) ;
m_context < < returnLabel ;
}
void Compiler : : appendConstructorCall ( FunctionDefinition const & _constructor )
{
eth : : AssemblyItem returnTag = m_context . pushNewTag ( ) ;
@ -107,11 +159,12 @@ void Compiler::appendConstructorCall(FunctionDefinition const& _constructor)
m_context < < returnTag ;
}
set < FunctionDefinition const * > Compiler : : getFunctionsNeededByConstructor ( FunctionDefinition const & _constructor )
set < FunctionDefinition const * > Compiler : : getFunctionsCalled ( set < ASTNode const * > const & _nodes )
{
// TODO this does not add virtual functions
CallGraph callgraph ;
callgraph . addFunction ( _constructor ) ;
callgraph . computeCallGraph ( ) ;
for ( ASTNode const * node : _nodes )
callgraph . addNode ( * node ) ;
return callgraph . getCalls ( ) ;
}
@ -193,8 +246,8 @@ void Compiler::appendReturnValuePacker(FunctionDefinition const& _function)
void Compiler : : registerStateVariables ( ContractDefinition const & _contract )
{
//@todo sort them?
for ( ASTPointer < VariableDeclaration > const & variable : _contract . getStateVariables ( ) )
for ( ContractDefinition const * contract : boost : : adaptors : : reverse ( _contract . getLinearizedBaseContracts ( ) ) )
for ( ASTPointer < VariableDeclaration > const & variable : contract - > getStateVariables ( ) )
m_context . addStateVariable ( * variable ) ;
}