|
@ -58,7 +58,10 @@ void Compiler::compileContract(ContractDefinition const& _contract, |
|
|
while (!functions.empty()) |
|
|
while (!functions.empty()) |
|
|
{ |
|
|
{ |
|
|
for (Declaration const* function: functions) |
|
|
for (Declaration const* function: functions) |
|
|
|
|
|
{ |
|
|
|
|
|
m_context.setStackOffset(0); |
|
|
function->accept(*this); |
|
|
function->accept(*this); |
|
|
|
|
|
} |
|
|
functions = m_context.getFunctionsWithoutCode(); |
|
|
functions = m_context.getFunctionsWithoutCode(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@ -79,37 +82,38 @@ void Compiler::initializeContext(ContractDefinition const& _contract, |
|
|
|
|
|
|
|
|
void Compiler::packIntoContractCreator(ContractDefinition const& _contract, CompilerContext const& _runtimeContext) |
|
|
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; |
|
|
|
|
|
|
|
|
|
|
|
// Determine the arguments that are used for the base constructors.
|
|
|
// Determine the arguments that are used for the base constructors.
|
|
|
std::vector<ContractDefinition const*> const& bases = _contract.getLinearizedBaseContracts(); |
|
|
std::vector<ContractDefinition const*> const& bases = _contract.getLinearizedBaseContracts(); |
|
|
for (ContractDefinition const* contract: bases) |
|
|
for (ContractDefinition const* contract: bases) |
|
|
|
|
|
{ |
|
|
|
|
|
if (FunctionDefinition const* constructor = contract->getConstructor()) |
|
|
|
|
|
for (auto const& modifier: constructor->getModifiers()) |
|
|
|
|
|
{ |
|
|
|
|
|
auto baseContract = dynamic_cast<ContractDefinition const*>( |
|
|
|
|
|
modifier->getName()->getReferencedDeclaration()); |
|
|
|
|
|
if (baseContract) |
|
|
|
|
|
if (m_baseArguments.count(baseContract->getConstructor()) == 0) |
|
|
|
|
|
m_baseArguments[baseContract->getConstructor()] = &modifier->getArguments(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
for (ASTPointer<InheritanceSpecifier> const& base: contract->getBaseContracts()) |
|
|
for (ASTPointer<InheritanceSpecifier> const& base: contract->getBaseContracts()) |
|
|
{ |
|
|
{ |
|
|
ContractDefinition const* baseContract = dynamic_cast<ContractDefinition const*>( |
|
|
ContractDefinition const* baseContract = dynamic_cast<ContractDefinition const*>( |
|
|
base->getName()->getReferencedDeclaration()); |
|
|
base->getName()->getReferencedDeclaration()); |
|
|
solAssert(baseContract, ""); |
|
|
solAssert(baseContract, ""); |
|
|
if (baseArguments.count(baseContract) == 0) |
|
|
|
|
|
baseArguments[baseContract] = &base->getArguments(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Call constructors in base-to-derived order.
|
|
|
if (m_baseArguments.count(baseContract->getConstructor()) == 0) |
|
|
// The Constructor for the most derived contract is called later.
|
|
|
m_baseArguments[baseContract->getConstructor()] = &base->getArguments(); |
|
|
for (unsigned i = 1; i < bases.size(); i++) |
|
|
} |
|
|
{ |
|
|
|
|
|
ContractDefinition const* base = bases[bases.size() - i]; |
|
|
|
|
|
solAssert(base, ""); |
|
|
|
|
|
initializeStateVariables(*base); |
|
|
|
|
|
FunctionDefinition const* baseConstructor = base->getConstructor(); |
|
|
|
|
|
if (!baseConstructor) |
|
|
|
|
|
continue; |
|
|
|
|
|
solAssert(baseArguments[base], ""); |
|
|
|
|
|
appendBaseConstructorCall(*baseConstructor, *baseArguments[base]); |
|
|
|
|
|
} |
|
|
} |
|
|
initializeStateVariables(_contract); |
|
|
// Initialization of state variables in base-to-derived order.
|
|
|
if (_contract.getConstructor()) |
|
|
for (ContractDefinition const* contract: boost::adaptors::reverse(bases)) |
|
|
appendConstructorCall(*_contract.getConstructor()); |
|
|
initializeStateVariables(*contract); |
|
|
|
|
|
|
|
|
|
|
|
if (FunctionDefinition const* constructor = _contract.getConstructor()) |
|
|
|
|
|
appendConstructor(*constructor); |
|
|
|
|
|
else if (auto c = m_context.getNextConstructor(_contract)) |
|
|
|
|
|
appendBaseConstructor(*c); |
|
|
|
|
|
|
|
|
eth::AssemblyItem sub = m_context.addSubroutine(_runtimeContext.getAssembly()); |
|
|
eth::AssemblyItem sub = m_context.addSubroutine(_runtimeContext.getAssembly()); |
|
|
// stack contains sub size
|
|
|
// stack contains sub size
|
|
@ -126,22 +130,23 @@ void Compiler::packIntoContractCreator(ContractDefinition const& _contract, Comp |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void Compiler::appendBaseConstructorCall(FunctionDefinition const& _constructor, |
|
|
void Compiler::appendBaseConstructor(FunctionDefinition const& _constructor) |
|
|
vector<ASTPointer<Expression>> const& _arguments) |
|
|
|
|
|
{ |
|
|
{ |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &_constructor); |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &_constructor); |
|
|
FunctionType constructorType(_constructor); |
|
|
FunctionType constructorType(_constructor); |
|
|
eth::AssemblyItem returnLabel = m_context.pushNewTag(); |
|
|
if (!constructorType.getParameterTypes().empty()) |
|
|
for (unsigned i = 0; i < _arguments.size(); ++i) |
|
|
{ |
|
|
compileExpression(*_arguments[i], constructorType.getParameterTypes()[i]); |
|
|
std::vector<ASTPointer<Expression>> const* arguments = m_baseArguments[&_constructor]; |
|
|
m_context.appendJumpTo(m_context.getFunctionEntryLabel(_constructor)); |
|
|
solAssert(arguments, ""); |
|
|
m_context << returnLabel; |
|
|
for (unsigned i = 0; i < arguments->size(); ++i) |
|
|
|
|
|
compileExpression(*(arguments->at(i)), constructorType.getParameterTypes()[i]); |
|
|
|
|
|
} |
|
|
|
|
|
_constructor.accept(*this); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void Compiler::appendConstructorCall(FunctionDefinition const& _constructor) |
|
|
void Compiler::appendConstructor(FunctionDefinition const& _constructor) |
|
|
{ |
|
|
{ |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &_constructor); |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &_constructor); |
|
|
eth::AssemblyItem returnTag = m_context.pushNewTag(); |
|
|
|
|
|
// copy constructor arguments from code to memory and then to stack, they are supplied after the actual program
|
|
|
// copy constructor arguments from code to memory and then to stack, they are supplied after the actual program
|
|
|
unsigned argumentSize = 0; |
|
|
unsigned argumentSize = 0; |
|
|
for (ASTPointer<VariableDeclaration> const& var: _constructor.getParameters()) |
|
|
for (ASTPointer<VariableDeclaration> const& var: _constructor.getParameters()) |
|
@ -155,8 +160,7 @@ void Compiler::appendConstructorCall(FunctionDefinition const& _constructor) |
|
|
m_context << eth::Instruction::CODECOPY; |
|
|
m_context << eth::Instruction::CODECOPY; |
|
|
appendCalldataUnpacker(FunctionType(_constructor).getParameterTypes(), true); |
|
|
appendCalldataUnpacker(FunctionType(_constructor).getParameterTypes(), true); |
|
|
} |
|
|
} |
|
|
m_context.appendJumpTo(m_context.getFunctionEntryLabel(_constructor)); |
|
|
_constructor.accept(*this); |
|
|
m_context << returnTag; |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void Compiler::appendFunctionSelector(ContractDefinition const& _contract) |
|
|
void Compiler::appendFunctionSelector(ContractDefinition const& _contract) |
|
@ -296,28 +300,36 @@ bool Compiler::visit(FunctionDefinition const& _function) |
|
|
// although note that this reduces the size of the visible stack
|
|
|
// although note that this reduces the size of the visible stack
|
|
|
|
|
|
|
|
|
m_context.startFunction(_function); |
|
|
m_context.startFunction(_function); |
|
|
m_returnTag = m_context.newTag(); |
|
|
|
|
|
m_breakTags.clear(); |
|
|
|
|
|
m_continueTags.clear(); |
|
|
|
|
|
m_stackCleanupForReturn = 0; |
|
|
|
|
|
m_currentFunction = &_function; |
|
|
|
|
|
m_modifierDepth = 0; |
|
|
|
|
|
|
|
|
|
|
|
// stack upon entry: [return address] [arg0] [arg1] ... [argn]
|
|
|
// stack upon entry: [return address] [arg0] [arg1] ... [argn]
|
|
|
// reserve additional slots: [retarg0] ... [retargm] [localvar0] ... [localvarp]
|
|
|
// reserve additional slots: [retarg0] ... [retargm] [localvar0] ... [localvarp]
|
|
|
|
|
|
|
|
|
unsigned parametersSize = CompilerUtils::getSizeOnStack(_function.getParameters()); |
|
|
unsigned parametersSize = CompilerUtils::getSizeOnStack(_function.getParameters()); |
|
|
m_context.adjustStackOffset(parametersSize); |
|
|
if (!_function.isConstructor()) |
|
|
|
|
|
// adding 1 for return address.
|
|
|
|
|
|
m_context.adjustStackOffset(parametersSize + 1); |
|
|
for (ASTPointer<VariableDeclaration const> const& variable: _function.getParameters()) |
|
|
for (ASTPointer<VariableDeclaration const> const& variable: _function.getParameters()) |
|
|
{ |
|
|
{ |
|
|
m_context.addVariable(*variable, parametersSize); |
|
|
m_context.addVariable(*variable, parametersSize); |
|
|
parametersSize -= variable->getType()->getSizeOnStack(); |
|
|
parametersSize -= variable->getType()->getSizeOnStack(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
for (ASTPointer<VariableDeclaration const> const& variable: _function.getReturnParameters()) |
|
|
for (ASTPointer<VariableDeclaration const> const& variable: _function.getReturnParameters()) |
|
|
m_context.addAndInitializeVariable(*variable); |
|
|
m_context.addAndInitializeVariable(*variable); |
|
|
for (VariableDeclaration const* localVariable: _function.getLocalVariables()) |
|
|
for (VariableDeclaration const* localVariable: _function.getLocalVariables()) |
|
|
m_context.addAndInitializeVariable(*localVariable); |
|
|
m_context.addAndInitializeVariable(*localVariable); |
|
|
|
|
|
|
|
|
|
|
|
if (_function.isConstructor()) |
|
|
|
|
|
if (auto c = m_context.getNextConstructor(dynamic_cast<ContractDefinition const&>(*_function.getScope()))) |
|
|
|
|
|
appendBaseConstructor(*c); |
|
|
|
|
|
|
|
|
|
|
|
m_returnTag = m_context.newTag(); |
|
|
|
|
|
m_breakTags.clear(); |
|
|
|
|
|
m_continueTags.clear(); |
|
|
|
|
|
m_stackCleanupForReturn = 0; |
|
|
|
|
|
m_currentFunction = &_function; |
|
|
|
|
|
m_modifierDepth = 0; |
|
|
|
|
|
|
|
|
appendModifierOrFunctionCode(); |
|
|
appendModifierOrFunctionCode(); |
|
|
|
|
|
|
|
|
m_context << m_returnTag; |
|
|
m_context << m_returnTag; |
|
@ -352,8 +364,14 @@ bool Compiler::visit(FunctionDefinition const& _function) |
|
|
} |
|
|
} |
|
|
//@todo assert that everything is in place now
|
|
|
//@todo assert that everything is in place now
|
|
|
|
|
|
|
|
|
m_context << eth::Instruction::JUMP; |
|
|
for (ASTPointer<VariableDeclaration const> const& variable: _function.getParameters() + _function.getReturnParameters()) |
|
|
|
|
|
m_context.removeVariable(*variable); |
|
|
|
|
|
for (VariableDeclaration const* localVariable: _function.getLocalVariables()) |
|
|
|
|
|
m_context.removeVariable(*localVariable); |
|
|
|
|
|
|
|
|
|
|
|
m_context.adjustStackOffset(-c_returnValuesSize); |
|
|
|
|
|
if (!_function.isConstructor()) |
|
|
|
|
|
m_context << eth::Instruction::JUMP; |
|
|
return false; |
|
|
return false; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@ -515,6 +533,16 @@ void Compiler::appendModifierOrFunctionCode() |
|
|
else |
|
|
else |
|
|
{ |
|
|
{ |
|
|
ASTPointer<ModifierInvocation> const& modifierInvocation = m_currentFunction->getModifiers()[m_modifierDepth]; |
|
|
ASTPointer<ModifierInvocation> const& modifierInvocation = m_currentFunction->getModifiers()[m_modifierDepth]; |
|
|
|
|
|
|
|
|
|
|
|
// constructor call should be excluded
|
|
|
|
|
|
if (dynamic_cast<ContractDefinition const*>(modifierInvocation->getName()->getReferencedDeclaration())) |
|
|
|
|
|
{ |
|
|
|
|
|
++m_modifierDepth; |
|
|
|
|
|
appendModifierOrFunctionCode(); |
|
|
|
|
|
--m_modifierDepth; |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
ModifierDefinition const& modifier = m_context.getFunctionModifier(modifierInvocation->getName()->getName()); |
|
|
ModifierDefinition const& modifier = m_context.getFunctionModifier(modifierInvocation->getName()->getName()); |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &modifier); |
|
|
CompilerContext::LocationSetter locationSetter(m_context, &modifier); |
|
|
solAssert(modifier.getParameters().size() == modifierInvocation->getArguments().size(), ""); |
|
|
solAssert(modifier.getParameters().size() == modifierInvocation->getArguments().size(), ""); |
|
|