aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLiana Husikyan <liana@ethdev.com>2015-02-28 00:41:22 +0800
committerLiana Husikyan <liana@ethdev.com>2015-03-02 21:28:08 +0800
commite9238e15b4058bf9b84c3c9c59720f255843de75 (patch)
treea3dab94130e93a6223f86f31f623b6089b5e07d8
parent32e210eccad6ea951feaa3441a6aefb9938f8a90 (diff)
downloaddexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar.gz
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar.bz2
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar.lz
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar.xz
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.tar.zst
dexon-solidity-e9238e15b4058bf9b84c3c9c59720f255843de75.zip
Implemented passing arguments to the base constructor.
-rw-r--r--AST.cpp31
-rw-r--r--AST.h5
-rw-r--r--Compiler.cpp108
-rw-r--r--Compiler.h7
-rw-r--r--CompilerContext.cpp33
-rw-r--r--CompilerContext.h8
6 files changed, 132 insertions, 60 deletions
diff --git a/AST.cpp b/AST.cpp
index 4c6db6a2..00660c5d 100644
--- a/AST.cpp
+++ b/AST.cpp
@@ -308,7 +308,9 @@ void FunctionDefinition::checkTypeRequirements()
if (!var->getType()->canLiveOutsideStorage())
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
- modifier->checkTypeRequirements();
+ modifier->checkTypeRequirements(isConstructor() ?
+ dynamic_cast<ContractDefinition const&>(*getScope()).getBaseContracts() :
+ vector<ASTPointer<InheritanceSpecifier>>());
m_body->checkTypeRequirements();
}
@@ -351,19 +353,34 @@ void ModifierDefinition::checkTypeRequirements()
m_body->checkTypeRequirements();
}
-void ModifierInvocation::checkTypeRequirements()
+void ModifierInvocation::checkTypeRequirements(std::vector<ASTPointer<InheritanceSpecifier>> const& _bases)
{
m_modifierName->checkTypeRequirements();
for (ASTPointer<Expression> const& argument: m_arguments)
argument->checkTypeRequirements();
- ModifierDefinition const* modifier = dynamic_cast<ModifierDefinition const*>(m_modifierName->getReferencedDeclaration());
- solAssert(modifier, "Function modifier not found.");
- vector<ASTPointer<VariableDeclaration>> const& parameters = modifier->getParameters();
- if (parameters.size() != m_arguments.size())
+ auto declaration = m_modifierName->getReferencedDeclaration();
+ vector<ASTPointer<VariableDeclaration>> emptyParameterList;
+ vector<ASTPointer<VariableDeclaration>> const* parameters = nullptr;
+ if (auto modifier = dynamic_cast<ModifierDefinition const*>(declaration))
+ parameters = &modifier->getParameters();
+ else
+ for (auto const& base: _bases)
+ if (declaration == base->getName()->getReferencedDeclaration())
+ {
+ m_referencedConstructor = dynamic_cast<ContractDefinition const&>(*declaration).getConstructor();
+ if (m_referencedConstructor)
+ parameters = &m_referencedConstructor->getParameters();
+ else
+ parameters = &emptyParameterList;
+ break;
+ }
+ if (!parameters)
+ BOOST_THROW_EXCEPTION(createTypeError("Referenced declaration is neither modifier nor base class."));
+ if (parameters->size() != m_arguments.size())
BOOST_THROW_EXCEPTION(createTypeError("Wrong argument count for modifier invocation."));
for (size_t i = 0; i < m_arguments.size(); ++i)
- if (!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*parameters[i]->getType()))
+ if (!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*(*parameters)[i]->getType()))
BOOST_THROW_EXCEPTION(createTypeError("Invalid type for argument in modifier invocation."));
}
diff --git a/AST.h b/AST.h
index d028e8a7..967c6792 100644
--- a/AST.h
+++ b/AST.h
@@ -510,7 +510,7 @@ private:
};
/**
- * Invocation/usage of a modifier in a function header.
+ * Invocation/usage of a modifier in a function header or a base constructor call.
*/
class ModifierInvocation: public ASTNode
{
@@ -525,11 +525,12 @@ public:
ASTPointer<Identifier> const& getName() const { return m_modifierName; }
std::vector<ASTPointer<Expression>> const& getArguments() const { return m_arguments; }
- void checkTypeRequirements();
+ void checkTypeRequirements(std::vector<ASTPointer<InheritanceSpecifier>> const& _bases);
private:
ASTPointer<Identifier> m_modifierName;
std::vector<ASTPointer<Expression>> m_arguments;
+ FunctionDefinition const* m_referencedConstructor = nullptr;
};
/**
diff --git a/Compiler.cpp b/Compiler.cpp
index 2f75d2ea..580fd5b6 100644
--- a/Compiler.cpp
+++ b/Compiler.cpp
@@ -58,7 +58,10 @@ void Compiler::compileContract(ContractDefinition const& _contract,
while (!functions.empty())
{
for (Declaration const* function: functions)
+ {
+ m_context.setStackOffset(0);
function->accept(*this);
+ }
functions = m_context.getFunctionsWithoutCode();
}
@@ -79,37 +82,38 @@ void Compiler::initializeContext(ContractDefinition const& _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;
-
// Determine the arguments that are used for the base constructors.
std::vector<ContractDefinition const*> const& bases = _contract.getLinearizedBaseContracts();
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())
{
ContractDefinition const* baseContract = dynamic_cast<ContractDefinition const*>(
base->getName()->getReferencedDeclaration());
solAssert(baseContract, "");
- if (baseArguments.count(baseContract) == 0)
- baseArguments[baseContract] = &base->getArguments();
- }
- // 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, "");
- initializeStateVariables(*base);
- FunctionDefinition const* baseConstructor = base->getConstructor();
- if (!baseConstructor)
- continue;
- solAssert(baseArguments[base], "");
- appendBaseConstructorCall(*baseConstructor, *baseArguments[base]);
+ if (m_baseArguments.count(baseContract->getConstructor()) == 0)
+ m_baseArguments[baseContract->getConstructor()] = &base->getArguments();
+ }
}
- initializeStateVariables(_contract);
- if (_contract.getConstructor())
- appendConstructorCall(*_contract.getConstructor());
+ // Initialization of state variables in base-to-derived order.
+ for (ContractDefinition const* contract: boost::adaptors::reverse(bases))
+ 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());
// stack contains sub size
@@ -126,22 +130,23 @@ void Compiler::packIntoContractCreator(ContractDefinition const& _contract, Comp
}
}
-void Compiler::appendBaseConstructorCall(FunctionDefinition const& _constructor,
- vector<ASTPointer<Expression>> const& _arguments)
+void Compiler::appendBaseConstructor(FunctionDefinition const& _constructor)
{
CompilerContext::LocationSetter locationSetter(m_context, &_constructor);
FunctionType constructorType(_constructor);
- eth::AssemblyItem returnLabel = m_context.pushNewTag();
- for (unsigned i = 0; i < _arguments.size(); ++i)
- compileExpression(*_arguments[i], constructorType.getParameterTypes()[i]);
- m_context.appendJumpTo(m_context.getFunctionEntryLabel(_constructor));
- m_context << returnLabel;
+ if (!constructorType.getParameterTypes().empty())
+ {
+ std::vector<ASTPointer<Expression>> const* arguments = m_baseArguments[&_constructor];
+ solAssert(arguments, "");
+ 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);
- eth::AssemblyItem returnTag = m_context.pushNewTag();
// copy constructor arguments from code to memory and then to stack, they are supplied after the actual program
unsigned argumentSize = 0;
for (ASTPointer<VariableDeclaration> const& var: _constructor.getParameters())
@@ -155,8 +160,7 @@ void Compiler::appendConstructorCall(FunctionDefinition const& _constructor)
m_context << eth::Instruction::CODECOPY;
appendCalldataUnpacker(FunctionType(_constructor).getParameterTypes(), true);
}
- m_context.appendJumpTo(m_context.getFunctionEntryLabel(_constructor));
- m_context << returnTag;
+ _constructor.accept(*this);
}
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
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]
// reserve additional slots: [retarg0] ... [retargm] [localvar0] ... [localvarp]
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())
{
m_context.addVariable(*variable, parametersSize);
parametersSize -= variable->getType()->getSizeOnStack();
}
+
for (ASTPointer<VariableDeclaration const> const& variable: _function.getReturnParameters())
m_context.addAndInitializeVariable(*variable);
for (VariableDeclaration const* localVariable: _function.getLocalVariables())
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();
m_context << m_returnTag;
@@ -352,8 +364,14 @@ bool Compiler::visit(FunctionDefinition const& _function)
}
//@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;
}
@@ -515,6 +533,16 @@ void Compiler::appendModifierOrFunctionCode()
else
{
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());
CompilerContext::LocationSetter locationSetter(m_context, &modifier);
solAssert(modifier.getParameters().size() == modifierInvocation->getArguments().size(), "");
diff --git a/Compiler.h b/Compiler.h
index 3ad2d8c6..2804e8ec 100644
--- a/Compiler.h
+++ b/Compiler.h
@@ -55,9 +55,8 @@ private:
/// Adds the code that is run at creation time. Should be run after exchanging the run-time context
/// with a new and initialized context. Adds the constructor code.
void packIntoContractCreator(ContractDefinition const& _contract, CompilerContext const& _runtimeContext);
- void appendBaseConstructorCall(FunctionDefinition const& _constructor,
- std::vector<ASTPointer<Expression>> const& _arguments);
- void appendConstructorCall(FunctionDefinition const& _constructor);
+ void appendBaseConstructor(FunctionDefinition const& _constructor);
+ void appendConstructor(FunctionDefinition const& _constructor);
void appendFunctionSelector(ContractDefinition const& _contract);
/// Creates code that unpacks the arguments for the given function represented by a vector of TypePointers.
/// From memory if @a _fromMemory is true, otherwise from call data.
@@ -94,6 +93,8 @@ private:
unsigned m_modifierDepth = 0;
FunctionDefinition const* m_currentFunction;
unsigned m_stackCleanupForReturn; ///< this number of stack elements need to be removed before jump to m_returnTag
+ // arguments for base constructors, filled in derived-to-base order
+ std::map<FunctionDefinition const*, std::vector<ASTPointer<Expression>> const*> m_baseArguments;
};
}
diff --git a/CompilerContext.cpp b/CompilerContext.cpp
index 18be337f..7184dc37 100644
--- a/CompilerContext.cpp
+++ b/CompilerContext.cpp
@@ -51,8 +51,6 @@ void CompilerContext::addStateVariable(VariableDeclaration const& _declaration)
void CompilerContext::startFunction(Declaration const& _function)
{
m_functionsWithCode.insert(&_function);
- m_localVariables.clear();
- m_asm.setDeposit(0);
*this << getFunctionEntryLabel(_function);
}
@@ -63,6 +61,12 @@ void CompilerContext::addVariable(VariableDeclaration const& _declaration,
m_localVariables[&_declaration] = unsigned(m_asm.deposit()) - _offsetToCurrent;
}
+void CompilerContext::removeVariable(VariableDeclaration const& _declaration)
+{
+ solAssert(m_localVariables.count(&_declaration), "");
+ m_localVariables.erase(&_declaration);
+}
+
void CompilerContext::addAndInitializeVariable(VariableDeclaration const& _declaration)
{
LocationSetter locationSetter(*this, &_declaration);
@@ -110,11 +114,8 @@ eth::AssemblyItem CompilerContext::getVirtualFunctionEntryLabel(FunctionDefiniti
eth::AssemblyItem CompilerContext::getSuperFunctionEntryLabel(string const& _name, ContractDefinition const& _base)
{
- // search for first contract after _base
- solAssert(!m_inheritanceHierarchy.empty(), "No inheritance hierarchy set.");
- auto it = find(m_inheritanceHierarchy.begin(), m_inheritanceHierarchy.end(), &_base);
- solAssert(it != m_inheritanceHierarchy.end(), "Base not found in inheritance hierarchy.");
- for (++it; it != m_inheritanceHierarchy.end(); ++it)
+ auto it = getSuperContract(_base);
+ for (; it != m_inheritanceHierarchy.end(); ++it)
for (ASTPointer<FunctionDefinition> const& function: (*it)->getDefinedFunctions())
if (!function->isConstructor() && function->getName() == _name)
return getFunctionEntryLabel(*function);
@@ -122,6 +123,16 @@ eth::AssemblyItem CompilerContext::getSuperFunctionEntryLabel(string const& _nam
return m_asm.newTag(); // not reached
}
+FunctionDefinition const* CompilerContext::getNextConstructor(ContractDefinition const& _contract) const
+{
+ vector<ContractDefinition const*>::const_iterator it = getSuperContract(_contract);
+ for (; it != m_inheritanceHierarchy.end(); it = getSuperContract(**it))
+ if ((*it)->getConstructor())
+ return (*it)->getConstructor();
+
+ return nullptr;
+}
+
set<Declaration const*> CompilerContext::getFunctionsWithoutCode()
{
set<Declaration const*> functions;
@@ -201,5 +212,13 @@ CompilerContext& CompilerContext::operator<<(bytes const& _data)
return *this;
}
+vector<ContractDefinition const*>::const_iterator CompilerContext::getSuperContract(ContractDefinition const& _contract) const
+{
+ solAssert(!m_inheritanceHierarchy.empty(), "No inheritance hierarchy set.");
+ auto it = find(m_inheritanceHierarchy.begin(), m_inheritanceHierarchy.end(), &_contract);
+ solAssert(it != m_inheritanceHierarchy.end(), "Base not found in inheritance hierarchy.");
+ return ++it;
+}
+
}
}
diff --git a/CompilerContext.h b/CompilerContext.h
index 94d6443e..301ef146 100644
--- a/CompilerContext.h
+++ b/CompilerContext.h
@@ -43,11 +43,13 @@ public:
void addMagicGlobal(MagicVariableDeclaration const& _declaration);
void addStateVariable(VariableDeclaration const& _declaration);
void addVariable(VariableDeclaration const& _declaration, unsigned _offsetToCurrent = 0);
+ void removeVariable(VariableDeclaration const& _declaration);
void addAndInitializeVariable(VariableDeclaration const& _declaration);
void setCompiledContracts(std::map<ContractDefinition const*, bytes const*> const& _contracts) { m_compiledContracts = _contracts; }
bytes const& getCompiledContract(ContractDefinition const& _contract) const;
+ void setStackOffset(int _offset) { m_asm.setDeposit(_offset); }
void adjustStackOffset(int _adjustment) { m_asm.adjustDeposit(_adjustment); }
unsigned getStackHeight() const { solAssert(m_asm.deposit() >= 0, ""); return unsigned(m_asm.deposit()); }
@@ -62,6 +64,8 @@ public:
/// @returns the entry label of function with the given name from the most derived class just
/// above _base in the current inheritance hierarchy.
eth::AssemblyItem getSuperFunctionEntryLabel(std::string const& _name, ContractDefinition const& _base);
+ FunctionDefinition const* getNextConstructor(ContractDefinition const& _contract) const;
+
/// @returns the set of functions for which we still need to generate code
std::set<Declaration const*> getFunctionsWithoutCode();
/// Resets function specific members, inserts the function entry label and marks the function
@@ -126,9 +130,11 @@ public:
LocationSetter(CompilerContext& _compilerContext, ASTNode const* _node):
ScopeGuard(std::bind(&CompilerContext::popVisitedNodes, _compilerContext)) { _compilerContext.pushVisitedNodes(_node); }
};
- eth::Assembly m_asm;
+
private:
+ std::vector<ContractDefinition const*>::const_iterator getSuperContract(const ContractDefinition &_contract) const;
+ eth::Assembly m_asm;
/// Magic global variables like msg, tx or this, distinguished by type.
std::set<Declaration const*> m_magicGlobals;
/// Other already compiled contracts to be used in contract creation calls.