aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLefteris Karapetsas <lefteris@refu.co>2015-04-02 21:10:35 +0800
committerLefteris Karapetsas <lefteris@refu.co>2015-04-17 21:27:31 +0800
commitd997dc55d154c93af01175d880734b9e737d34ca (patch)
tree3e08f516eb4fdb499223e681b93fd86504558589
parent6e5de4832da7a3c2fdbc87b71523ac671083d9c0 (diff)
downloaddexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar.gz
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar.bz2
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar.lz
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar.xz
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.tar.zst
dexon-solidity-d997dc55d154c93af01175d880734b9e737d34ca.zip
Allowing abstract contracts constructor to have no args
- If a constructor is part of an abstract contract we can omit its arguments - IF a contract is abstract make sure to not create and/or request Assembly code about it since it's not compiled
-rw-r--r--AST.cpp14
-rw-r--r--AST.h2
-rw-r--r--Compiler.cpp1
-rw-r--r--CompilerStack.cpp16
-rw-r--r--CompilerStack.h4
5 files changed, 22 insertions, 15 deletions
diff --git a/AST.cpp b/AST.cpp
index 0abdf781..37016646 100644
--- a/AST.cpp
+++ b/AST.cpp
@@ -281,7 +281,7 @@ void InheritanceSpecifier::checkTypeRequirements()
ContractDefinition const* base = dynamic_cast<ContractDefinition const*>(m_baseName->getReferencedDeclaration());
solAssert(base, "Base contract not available.");
TypePointers parameterTypes = ContractType(*base).getConstructorType()->getParameterTypes();
- if (parameterTypes.size() != m_arguments.size())
+ if (m_arguments.size() != 0 && parameterTypes.size() != m_arguments.size())
BOOST_THROW_EXCEPTION(createTypeError("Wrong argument count for constructor call."));
for (size_t i = 0; i < m_arguments.size(); ++i)
if (!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*parameterTypes[i]))
@@ -348,8 +348,8 @@ void FunctionDefinition::checkTypeRequirements()
}
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
modifier->checkTypeRequirements(isConstructor() ?
- dynamic_cast<ContractDefinition const&>(*getScope()).getBaseContracts() :
- vector<ASTPointer<InheritanceSpecifier>>());
+ dynamic_cast<ContractDefinition const&>(*getScope()).getLinearizedBaseContracts() :
+ vector<ContractDefinition const*>());
if (m_body)
m_body->checkTypeRequirements();
}
@@ -426,7 +426,7 @@ void ModifierDefinition::checkTypeRequirements()
m_body->checkTypeRequirements();
}
-void ModifierInvocation::checkTypeRequirements(vector<ASTPointer<InheritanceSpecifier>> const& _bases)
+void ModifierInvocation::checkTypeRequirements(vector<ContractDefinition const*> const& _bases)
{
m_modifierName->checkTypeRequirements();
for (ASTPointer<Expression> const& argument: m_arguments)
@@ -439,10 +439,10 @@ void ModifierInvocation::checkTypeRequirements(vector<ASTPointer<InheritanceSpec
parameters = &modifier->getParameters();
else
// check parameters for Base constructors
- for (auto const& base: _bases)
- if (declaration == base->getName()->getReferencedDeclaration())
+ for (auto const* base: _bases)
+ if (declaration == base)
{
- if (auto referencedConstructor = dynamic_cast<ContractDefinition const&>(*declaration).getConstructor())
+ if (auto referencedConstructor = base->getConstructor())
parameters = &referencedConstructor->getParameters();
else
parameters = &emptyParameterList;
diff --git a/AST.h b/AST.h
index 9f5f9fcb..8c36caea 100644
--- a/AST.h
+++ b/AST.h
@@ -566,7 +566,7 @@ public:
std::vector<ASTPointer<Expression>> const& getArguments() const { return m_arguments; }
/// @param _bases is the list of base contracts for base constructor calls. For modifiers an empty vector should be passed.
- void checkTypeRequirements(std::vector<ASTPointer<InheritanceSpecifier>> const& _bases);
+ void checkTypeRequirements(std::vector<ContractDefinition const*> const& _bases);
private:
ASTPointer<Identifier> m_modifierName;
diff --git a/Compiler.cpp b/Compiler.cpp
index 886565cb..f0197e23 100644
--- a/Compiler.cpp
+++ b/Compiler.cpp
@@ -136,6 +136,7 @@ void Compiler::appendBaseConstructor(FunctionDefinition const& _constructor)
FunctionType constructorType(_constructor);
if (!constructorType.getParameterTypes().empty())
{
+ solAssert(m_baseArguments.count(&_constructor), "");
std::vector<ASTPointer<Expression>> const* arguments = m_baseArguments[&_constructor];
solAssert(arguments, "");
for (unsigned i = 0; i < arguments->size(); ++i)
diff --git a/CompilerStack.cpp b/CompilerStack.cpp
index 1301bfa5..c35d9324 100644
--- a/CompilerStack.cpp
+++ b/CompilerStack.cpp
@@ -157,14 +157,16 @@ bytes const& CompilerStack::compile(string const& _sourceCode, bool _optimize)
return getBytecode();
}
-eth::AssemblyItems const& CompilerStack::getAssemblyItems(string const& _contractName) const
+eth::AssemblyItems const* CompilerStack::getAssemblyItems(string const& _contractName) const
{
- return getContract(_contractName).compiler->getAssemblyItems();
+ Contract const& contract = getContract(_contractName);
+ return contract.compiler ? &getContract(_contractName).compiler->getAssemblyItems() : nullptr;
}
-eth::AssemblyItems const& CompilerStack::getRuntimeAssemblyItems(string const& _contractName) const
+eth::AssemblyItems const* CompilerStack::getRuntimeAssemblyItems(string const& _contractName) const
{
- return getContract(_contractName).compiler->getRuntimeAssemblyItems();
+ Contract const& contract = getContract(_contractName);
+ return contract.compiler ? &getContract(_contractName).compiler->getRuntimeAssemblyItems() : nullptr;
}
bytes const& CompilerStack::getBytecode(string const& _contractName) const
@@ -184,7 +186,11 @@ dev::h256 CompilerStack::getContractCodeHash(string const& _contractName) const
void CompilerStack::streamAssembly(ostream& _outStream, string const& _contractName, StringMap _sourceCodes) const
{
- getContract(_contractName).compiler->streamAssembly(_outStream, _sourceCodes);
+ Contract const& contract = getContract(_contractName);
+ if (contract.compiler)
+ getContract(_contractName).compiler->streamAssembly(_outStream, _sourceCodes);
+ else
+ _outStream << "Contract not fully implemented" << endl;
}
string const& CompilerStack::getInterface(string const& _contractName) const
diff --git a/CompilerStack.h b/CompilerStack.h
index 1cf576ab..19c1ba4e 100644
--- a/CompilerStack.h
+++ b/CompilerStack.h
@@ -94,9 +94,9 @@ public:
/// @returns the runtime bytecode for the contract, i.e. the code that is returned by the constructor.
bytes const& getRuntimeBytecode(std::string const& _contractName = "") const;
/// @returns normal contract assembly items
- eth::AssemblyItems const& getAssemblyItems(std::string const& _contractName = "") const;
+ eth::AssemblyItems const* getAssemblyItems(std::string const& _contractName = "") const;
/// @returns runtime contract assembly items
- eth::AssemblyItems const& getRuntimeAssemblyItems(std::string const& _contractName = "") const;
+ eth::AssemblyItems const* getRuntimeAssemblyItems(std::string const& _contractName = "") const;
/// @returns hash of the runtime bytecode for the contract, i.e. the code that is returned by the constructor.
dev::h256 getContractCodeHash(std::string const& _contractName = "") const;