aboutsummaryrefslogtreecommitdiffstats
path: root/libsolidity/AST.cpp
diff options
context:
space:
mode:
authorchriseth <c@ethdev.com>2015-09-16 22:56:30 +0800
committerchriseth <c@ethdev.com>2015-09-22 02:03:05 +0800
commit34a81fd60e0218ef93813ffce22fb6d0ed4955e2 (patch)
treee15e3f2c224688c43859d4c4588743c8fd07adc1 /libsolidity/AST.cpp
parent352c196eb37744b8054909a9801d55c672d0eb1c (diff)
downloaddexon-solidity-34a81fd60e0218ef93813ffce22fb6d0ed4955e2.tar.gz
dexon-solidity-34a81fd60e0218ef93813ffce22fb6d0ed4955e2.tar.zst
dexon-solidity-34a81fd60e0218ef93813ffce22fb6d0ed4955e2.zip
Refactoring: Check types outside of AST and recover from some errors.
Diffstat (limited to 'libsolidity/AST.cpp')
-rw-r--r--libsolidity/AST.cpp1027
1 files changed, 31 insertions, 996 deletions
diff --git a/libsolidity/AST.cpp b/libsolidity/AST.cpp
index daa84016..4cd6cc0f 100644
--- a/libsolidity/AST.cpp
+++ b/libsolidity/AST.cpp
@@ -22,7 +22,6 @@
#include <algorithm>
#include <functional>
-#include <boost/range/adaptor/reversed.hpp>
#include <libsolidity/Utils.h>
#include <libsolidity/AST.h>
#include <libsolidity/ASTVisitor.h>
@@ -32,83 +31,21 @@
#include <libdevcore/SHA3.h>
using namespace std;
+using namespace dev;
+using namespace dev::solidity;
-namespace dev
+ASTNode::ASTNode(SourceLocation const& _location):
+ m_location(_location)
{
-namespace solidity
-{
-
-TypeError ASTNode::createTypeError(string const& _description) const
-{
- return TypeError() << errinfo_sourceLocation(location()) << errinfo_comment(_description);
}
-TypePointer ContractDefinition::type(ContractDefinition const* _currentContract) const
+ASTNode::~ASTNode()
{
- return make_shared<TypeType>(make_shared<ContractType>(*this), _currentContract);
}
-void ContractDefinition::checkTypeRequirements()
+TypeError ASTNode::createTypeError(string const& _description) const
{
- for (ASTPointer<InheritanceSpecifier> const& baseSpecifier: baseContracts())
- baseSpecifier->checkTypeRequirements();
-
- checkDuplicateFunctions();
- checkIllegalOverrides();
- checkAbstractFunctions();
- checkAbstractConstructors();
-
- FunctionDefinition const* functionDefinition = constructor();
- if (functionDefinition && !functionDefinition->returnParameters().empty())
- BOOST_THROW_EXCEPTION(functionDefinition->returnParameterList()->createTypeError(
- "Non-empty \"returns\" directive for constructor."
- ));
-
- FunctionDefinition const* fallbackFunction = nullptr;
- for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
- {
- if (function->name().empty())
- {
- if (fallbackFunction)
- BOOST_THROW_EXCEPTION(DeclarationError() << errinfo_comment("Only one fallback function is allowed."));
- else
- {
- fallbackFunction = function.get();
- if (!fallbackFunction->parameters().empty())
- BOOST_THROW_EXCEPTION(fallbackFunction->parameterList().createTypeError("Fallback function cannot take parameters."));
- }
- }
- if (!function->isFullyImplemented())
- setFullyImplemented(false);
- }
-
- for (ASTPointer<VariableDeclaration> const& variable: m_stateVariables)
- variable->checkTypeRequirements();
-
- for (ASTPointer<EventDefinition> const& event: events())
- event->checkTypeRequirements();
-
- for (ASTPointer<ModifierDefinition> const& modifier: functionModifiers())
- modifier->checkTypeRequirements();
-
- for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
- function->checkTypeRequirements();
-
- checkExternalTypeClashes();
- // check for hash collisions in function signatures
- set<FixedHash<4>> hashes;
- for (auto const& it: interfaceFunctionList())
- {
- FixedHash<4> const& hash = it.first;
- if (hashes.count(hash))
- BOOST_THROW_EXCEPTION(createTypeError(
- string("Function signature hash collision for ") + it.second->externalSignature()
- ));
- hashes.insert(hash);
- }
-
- if (isLibrary())
- checkLibraryRequirements();
+ return TypeError() << errinfo_sourceLocation(location()) << errinfo_comment(_description);
}
map<FixedHash<4>, FunctionTypePointer> ContractDefinition::interfaceFunctions() const
@@ -137,227 +74,20 @@ FunctionDefinition const* ContractDefinition::constructor() const
FunctionDefinition const* ContractDefinition::fallbackFunction() const
{
- for (ContractDefinition const* contract: linearizedBaseContracts())
+ for (ContractDefinition const* contract: annotation().linearizedBaseContracts)
for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
if (f->name().empty())
return f.get();
return nullptr;
}
-void ContractDefinition::checkDuplicateFunctions() const
-{
- /// Checks that two functions with the same name defined in this contract have different
- /// argument types and that there is at most one constructor.
- map<string, vector<FunctionDefinition const*>> functions;
- for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
- functions[function->name()].push_back(function.get());
-
- if (functions[name()].size() > 1)
- {
- SecondarySourceLocation ssl;
- auto it = functions[name()].begin();
- ++it;
- for (; it != functions[name()].end(); ++it)
- ssl.append("Another declaration is here:", (*it)->location());
-
- BOOST_THROW_EXCEPTION(
- DeclarationError() <<
- errinfo_sourceLocation(functions[name()].front()->location()) <<
- errinfo_comment("More than one constructor defined.") <<
- errinfo_secondarySourceLocation(ssl)
- );
- }
- for (auto const& it: functions)
- {
- vector<FunctionDefinition const*> const& overloads = it.second;
- for (size_t i = 0; i < overloads.size(); ++i)
- for (size_t j = i + 1; j < overloads.size(); ++j)
- if (FunctionType(*overloads[i]).hasEqualArgumentTypes(FunctionType(*overloads[j])))
- BOOST_THROW_EXCEPTION(
- DeclarationError() <<
- errinfo_sourceLocation(overloads[j]->location()) <<
- errinfo_comment("Function with same name and arguments defined twice.") <<
- errinfo_secondarySourceLocation(SecondarySourceLocation().append(
- "Other declaration is here:", overloads[i]->location())
- )
-
- );
- }
-}
-
-void ContractDefinition::checkAbstractFunctions()
-{
- // Mapping from name to function definition (exactly one per argument type equality class) and
- // flag to indicate whether it is fully implemented.
- using FunTypeAndFlag = std::pair<FunctionTypePointer, bool>;
- map<string, vector<FunTypeAndFlag>> functions;
-
- // Search from base to derived
- for (ContractDefinition const* contract: boost::adaptors::reverse(linearizedBaseContracts()))
- for (ASTPointer<FunctionDefinition> const& function: contract->definedFunctions())
- {
- auto& overloads = functions[function->name()];
- FunctionTypePointer funType = make_shared<FunctionType>(*function);
- auto it = find_if(overloads.begin(), overloads.end(), [&](FunTypeAndFlag const& _funAndFlag)
- {
- return funType->hasEqualArgumentTypes(*_funAndFlag.first);
- });
- if (it == overloads.end())
- overloads.push_back(make_pair(funType, function->isFullyImplemented()));
- else if (it->second)
- {
- if (!function->isFullyImplemented())
- BOOST_THROW_EXCEPTION(function->createTypeError("Redeclaring an already implemented function as abstract"));
- }
- else if (function->isFullyImplemented())
- it->second = true;
- }
-
- // Set to not fully implemented if at least one flag is false.
- for (auto const& it: functions)
- for (auto const& funAndFlag: it.second)
- if (!funAndFlag.second)
- {
- setFullyImplemented(false);
- return;
- }
-}
-
-void ContractDefinition::checkAbstractConstructors()
-{
- set<ContractDefinition const*> argumentsNeeded;
- // check that we get arguments for all base constructors that need it.
- // If not mark the contract as abstract (not fully implemented)
-
- vector<ContractDefinition const*> const& bases = linearizedBaseContracts();
- for (ContractDefinition const* contract: bases)
- if (FunctionDefinition const* constructor = contract->constructor())
- if (contract != this && !constructor->parameters().empty())
- argumentsNeeded.insert(contract);
-
- for (ContractDefinition const* contract: bases)
- {
- if (FunctionDefinition const* constructor = contract->constructor())
- for (auto const& modifier: constructor->modifiers())
- {
- auto baseContract = dynamic_cast<ContractDefinition const*>(
- &modifier->name()->referencedDeclaration()
- );
- if (baseContract)
- argumentsNeeded.erase(baseContract);
- }
-
-
- for (ASTPointer<InheritanceSpecifier> const& base: contract->baseContracts())
- {
- auto baseContract = dynamic_cast<ContractDefinition const*>(
- &base->name()->referencedDeclaration()
- );
- solAssert(baseContract, "");
- if (!base->arguments().empty())
- argumentsNeeded.erase(baseContract);
- }
- }
- if (!argumentsNeeded.empty())
- setFullyImplemented(false);
-}
-
-void ContractDefinition::checkIllegalOverrides() const
-{
- // TODO unify this at a later point. for this we need to put the constness and the access specifier
- // into the types
- map<string, vector<FunctionDefinition const*>> functions;
- map<string, ModifierDefinition const*> modifiers;
-
- // We search from derived to base, so the stored item causes the error.
- for (ContractDefinition const* contract: linearizedBaseContracts())
- {
- for (ASTPointer<FunctionDefinition> const& function: contract->definedFunctions())
- {
- if (function->isConstructor())
- continue; // constructors can neither be overridden nor override anything
- string const& name = function->name();
- if (modifiers.count(name))
- BOOST_THROW_EXCEPTION(modifiers[name]->createTypeError("Override changes function to modifier."));
- FunctionType functionType(*function);
- // function should not change the return type
- for (FunctionDefinition const* overriding: functions[name])
- {
- FunctionType overridingType(*overriding);
- if (!overridingType.hasEqualArgumentTypes(functionType))
- continue;
- if (
- overriding->visibility() != function->visibility() ||
- overriding->isDeclaredConst() != function->isDeclaredConst() ||
- overridingType != functionType
- )
- BOOST_THROW_EXCEPTION(overriding->createTypeError("Override changes extended function signature."));
- }
- functions[name].push_back(function.get());
- }
- for (ASTPointer<ModifierDefinition> const& modifier: contract->functionModifiers())
- {
- string const& name = modifier->name();
- ModifierDefinition const*& override = modifiers[name];
- if (!override)
- override = modifier.get();
- else if (ModifierType(*override) != ModifierType(*modifier))
- BOOST_THROW_EXCEPTION(override->createTypeError("Override changes modifier signature."));
- if (!functions[name].empty())
- BOOST_THROW_EXCEPTION(override->createTypeError("Override changes modifier to function."));
- }
- }
-}
-
-void ContractDefinition::checkExternalTypeClashes() const
-{
- map<string, vector<pair<Declaration const*, shared_ptr<FunctionType>>>> externalDeclarations;
- for (ContractDefinition const* contract: linearizedBaseContracts())
- {
- for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
- if (f->isPartOfExternalInterface())
- {
- auto functionType = make_shared<FunctionType>(*f);
- externalDeclarations[functionType->externalSignature(f->name())].push_back(
- make_pair(f.get(), functionType)
- );
- }
- for (ASTPointer<VariableDeclaration> const& v: contract->stateVariables())
- if (v->isPartOfExternalInterface())
- {
- auto functionType = make_shared<FunctionType>(*v);
- externalDeclarations[functionType->externalSignature(v->name())].push_back(
- make_pair(v.get(), functionType)
- );
- }
- }
- for (auto const& it: externalDeclarations)
- for (size_t i = 0; i < it.second.size(); ++i)
- for (size_t j = i + 1; j < it.second.size(); ++j)
- if (!it.second[i].second->hasEqualArgumentTypes(*it.second[j].second))
- BOOST_THROW_EXCEPTION(it.second[j].first->createTypeError(
- "Function overload clash during conversion to external types for arguments."
- ));
-}
-
-void ContractDefinition::checkLibraryRequirements() const
-{
- solAssert(m_isLibrary, "");
- if (!m_baseContracts.empty())
- BOOST_THROW_EXCEPTION(createTypeError("Library is not allowed to inherit."));
-
- for (auto const& var: m_stateVariables)
- if (!var->isConstant())
- BOOST_THROW_EXCEPTION(var->createTypeError("Library cannot have non-constant state variables"));
-}
-
vector<ASTPointer<EventDefinition>> const& ContractDefinition::interfaceEvents() const
{
if (!m_interfaceEvents)
{
set<string> eventsSeen;
m_interfaceEvents.reset(new vector<ASTPointer<EventDefinition>>());
- for (ContractDefinition const* contract: linearizedBaseContracts())
+ for (ContractDefinition const* contract: annotation().linearizedBaseContracts)
for (ASTPointer<EventDefinition> const& e: contract->events())
if (eventsSeen.count(e->name()) == 0)
{
@@ -375,7 +105,7 @@ vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::inter
set<string> functionsSeen;
set<string> signaturesSeen;
m_interfaceFunctionList.reset(new vector<pair<FixedHash<4>, FunctionTypePointer>>());
- for (ContractDefinition const* contract: linearizedBaseContracts())
+ for (ContractDefinition const* contract: annotation().linearizedBaseContracts)
{
for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
{
@@ -395,7 +125,7 @@ vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::inter
if (functionsSeen.count(v->name()) == 0 && v->isPartOfExternalInterface())
{
FunctionType ftype(*v);
- solAssert(v->type().get(), "");
+ solAssert(!!v->annotation().type.get(), "");
functionsSeen.insert(v->name());
FixedHash<4> hash(dev::sha3(ftype.externalSignature(v->name())));
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*v)));
@@ -453,45 +183,9 @@ vector<Declaration const*> const& ContractDefinition::inheritableMembers() const
return *m_inheritableMembers;
}
-TypePointer EnumValue::type(ContractDefinition const*) const
+TypePointer ContractDefinition::type(ContractDefinition const* m_currentContract) const
{
- EnumDefinition const* parentDef = dynamic_cast<EnumDefinition const*>(scope());
- solAssert(parentDef, "Enclosing Scope of EnumValue was not set");
- return make_shared<EnumType>(*parentDef);
-}
-
-void InheritanceSpecifier::checkTypeRequirements()
-{
- m_baseName->checkTypeRequirements(nullptr);
- for (ASTPointer<Expression> const& argument: m_arguments)
- argument->checkTypeRequirements(nullptr);
-
- ContractDefinition const* base = dynamic_cast<ContractDefinition const*>(&m_baseName->referencedDeclaration());
- solAssert(base, "Base contract not available.");
-
- if (base->isLibrary())
- BOOST_THROW_EXCEPTION(createTypeError("Libraries cannot be inherited from."));
-
- TypePointers parameterTypes = ContractType(*base).constructorType()->parameterTypes();
- if (!m_arguments.empty() && parameterTypes.size() != m_arguments.size())
- BOOST_THROW_EXCEPTION(createTypeError(
- "Wrong argument count for constructor call: " +
- toString(m_arguments.size()) +
- " arguments given but expected " +
- toString(parameterTypes.size()) +
- "."
- ));
-
- for (size_t i = 0; i < m_arguments.size(); ++i)
- if (!m_arguments[i]->type()->isImplicitlyConvertibleTo(*parameterTypes[i]))
- BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
- "Invalid type for argument in constructor call. "
- "Invalid implicit conversion from " +
- m_arguments[i]->type()->toString() +
- " to " +
- parameterTypes[i]->toString() +
- " requested."
- ));
+ return make_shared<TypeType>(make_shared<ContractType>(*this), m_currentContract);
}
TypePointer StructDefinition::type(ContractDefinition const*) const
@@ -499,38 +193,11 @@ TypePointer StructDefinition::type(ContractDefinition const*) const
return make_shared<TypeType>(make_shared<StructType>(*this));
}
-void StructDefinition::checkMemberTypes() const
-{
- for (ASTPointer<VariableDeclaration> const& member: members())
- if (!member->type()->canBeStored())
- BOOST_THROW_EXCEPTION(member->createTypeError("Type cannot be used in struct."));
-}
-
-void StructDefinition::checkRecursion() const
+TypePointer EnumValue::type(ContractDefinition const*) const
{
- using StructPointer = StructDefinition const*;
- using StructPointersSet = set<StructPointer>;
- function<void(StructPointer,StructPointersSet const&)> check = [&](StructPointer _struct, StructPointersSet const& _parents)
- {
- if (_parents.count(_struct))
- BOOST_THROW_EXCEPTION(
- ParserError() <<
- errinfo_sourceLocation(_struct->location()) <<
- errinfo_comment("Recursive struct definition.")
- );
- set<StructDefinition const*> parents = _parents;
- parents.insert(_struct);
- for (ASTPointer<VariableDeclaration> const& member: _struct->members())
- if (member->type()->category() == Type::Category::Struct)
- {
- auto const& typeName = dynamic_cast<UserDefinedTypeName const&>(*member->typeName());
- check(
- &dynamic_cast<StructDefinition const&>(*typeName.referencedDeclaration()),
- parents
- );
- }
- };
- check(this, StructPointersSet{});
+ auto parentDef = dynamic_cast<EnumDefinition const*>(scope());
+ solAssert(parentDef, "Enclosing Scope of EnumValue was not set");
+ return make_shared<EnumType>(*parentDef);
}
TypePointer EnumDefinition::type(ContractDefinition const*) const
@@ -543,92 +210,25 @@ TypePointer FunctionDefinition::type(ContractDefinition const*) const
return make_shared<FunctionType>(*this);
}
-void FunctionDefinition::checkTypeRequirements()
-{
- for (ASTPointer<VariableDeclaration> const& var: parameters() + returnParameters())
- {
- if (!var->type()->canLiveOutsideStorage())
- BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
- if (visibility() >= Visibility::Public && !(var->type()->externalType()))
- BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed for public and external functions."));
- }
- for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
- modifier->checkTypeRequirements(isConstructor() ?
- dynamic_cast<ContractDefinition const&>(*scope()).linearizedBaseContracts() :
- vector<ContractDefinition const*>());
- if (m_body)
- m_body->checkTypeRequirements();
-}
-
string FunctionDefinition::externalSignature() const
{
return FunctionType(*this).externalSignature(name());
}
-bool VariableDeclaration::isLValue() const
+TypePointer ModifierDefinition::type(ContractDefinition const*) const
{
- // External function parameters and constant declared variables are Read-Only
- return !isExternalCallableParameter() && !m_isConstant;
+ return make_shared<ModifierType>(*this);
}
-void VariableDeclaration::checkTypeRequirements()
+TypePointer EventDefinition::type(ContractDefinition const*) const
{
- // Variables can be declared without type (with "var"), in which case the first assignment
- // sets the type.
- // Note that assignments before the first declaration are legal because of the special scoping
- // rules inherited from JavaScript.
- if (m_isConstant)
- {
- if (!dynamic_cast<ContractDefinition const*>(scope()))
- BOOST_THROW_EXCEPTION(createTypeError("Illegal use of \"constant\" specifier."));
- if (!m_value)
- BOOST_THROW_EXCEPTION(createTypeError("Uninitialized \"constant\" variable."));
- if (m_type && !m_type->isValueType())
- {
- // TODO: const is implemented only for uint, bytesXX, string and enums types.
- bool constImplemented = false;
- if (auto arrayType = dynamic_cast<ArrayType const*>(m_type.get()))
- constImplemented = arrayType->isByteArray();
- if (!constImplemented)
- BOOST_THROW_EXCEPTION(createTypeError(
- "Illegal use of \"constant\" specifier. \"constant\" "
- "is not yet implemented for this type."
- ));
- }
- }
- if (m_type)
- {
- if (m_value)
- m_value->expectType(*m_type);
- }
- else
- {
- if (!m_value)
- // This feature might be extended in the future.
- BOOST_THROW_EXCEPTION(createTypeError("Assignment necessary for type detection."));
- m_value->checkTypeRequirements(nullptr);
+ return make_shared<FunctionType>(*this);
+}
- TypePointer const& type = m_value->type();
- if (
- type->category() == Type::Category::IntegerConstant &&
- !dynamic_pointer_cast<IntegerConstantType const>(type)->integerType()
- )
- BOOST_THROW_EXCEPTION(m_value->createTypeError("Invalid integer constant " + type->toString() + "."));
- else if (type->category() == Type::Category::Void)
- BOOST_THROW_EXCEPTION(createTypeError("Variable cannot have void type."));
- m_type = type->mobileType();
- }
- solAssert(!!m_type, "");
- if (!m_isStateVariable)
- {
- if (m_type->dataStoredIn(DataLocation::Memory) || m_type->dataStoredIn(DataLocation::CallData))
- if (!m_type->canLiveOutsideStorage())
- BOOST_THROW_EXCEPTION(createTypeError(
- "Type " + m_type->toString() + " is only valid in storage."
- ));
- }
- else if (visibility() >= Visibility::Public && !FunctionType(*this).externalType())
- BOOST_THROW_EXCEPTION(createTypeError("Internal type is not allowed for public state variables."));
+bool VariableDeclaration::isLValue() const
+{
+ // External function parameters and constant declared variables are Read-Only
+ return !isExternalCallableParameter() && !m_isConstant;
}
bool VariableDeclaration::isCallableParameter() const
@@ -657,578 +257,13 @@ bool VariableDeclaration::isExternalCallableParameter() const
return false;
}
-TypePointer ModifierDefinition::type(ContractDefinition const*) const
+bool VariableDeclaration::canHaveAutoType() const
{
- return make_shared<ModifierType>(*this);
-}
-
-void ModifierDefinition::checkTypeRequirements()
-{
- m_body->checkTypeRequirements();
-}
-
-void ModifierInvocation::checkTypeRequirements(vector<ContractDefinition const*> const& _bases)
-{
- TypePointers argumentTypes;
- for (ASTPointer<Expression> const& argument: m_arguments)
- {
- argument->checkTypeRequirements(nullptr);
- argumentTypes.push_back(argument->type());
- }
- m_modifierName->checkTypeRequirements(&argumentTypes);
-
- auto const* declaration = &m_modifierName->referencedDeclaration();
- vector<ASTPointer<VariableDeclaration>> emptyParameterList;
- vector<ASTPointer<VariableDeclaration>> const* parameters = nullptr;
- if (auto modifier = dynamic_cast<ModifierDefinition const*>(declaration))
- parameters = &modifier->parameters();
- else
- // check parameters for Base constructors
- for (ContractDefinition const* base: _bases)
- if (declaration == base)
- {
- if (auto referencedConstructor = base->constructor())
- parameters = &referencedConstructor->parameters();
- 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: " +
- toString(m_arguments.size()) +
- " arguments given but expected " +
- toString(parameters->size()) +
- "."
- ));
- for (size_t i = 0; i < m_arguments.size(); ++i)
- if (!m_arguments[i]->type()->isImplicitlyConvertibleTo(*(*parameters)[i]->type()))
- BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
- "Invalid type for argument in modifier invocation. "
- "Invalid implicit conversion from " +
- m_arguments[i]->type()->toString() +
- " to " +
- (*parameters)[i]->type()->toString() +
- " requested."
- ));
-}
-
-void EventDefinition::checkTypeRequirements()
-{
- int numIndexed = 0;
- for (ASTPointer<VariableDeclaration> const& var: parameters())
- {
- if (var->isIndexed())
- numIndexed++;
- if (numIndexed > 3)
- BOOST_THROW_EXCEPTION(createTypeError("More than 3 indexed arguments for event."));
- if (!var->type()->canLiveOutsideStorage())
- BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
- if (!var->type()->externalType())
- BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed as event parameter type."));
- }
-}
-
-void Block::checkTypeRequirements()
-{
- for (shared_ptr<Statement> const& statement: m_statements)
- statement->checkTypeRequirements();
-}
-
-void IfStatement::checkTypeRequirements()
-{
- m_condition->expectType(BoolType());
- m_trueBody->checkTypeRequirements();
- if (m_falseBody)
- m_falseBody->checkTypeRequirements();
-}
-
-void WhileStatement::checkTypeRequirements()
-{
- m_condition->expectType(BoolType());
- m_body->checkTypeRequirements();
-}
-
-void ForStatement::checkTypeRequirements()
-{
- if (m_initExpression)
- m_initExpression->checkTypeRequirements();
- if (m_condExpression)
- m_condExpression->expectType(BoolType());
- if (m_loopExpression)
- m_loopExpression->checkTypeRequirements();
- m_body->checkTypeRequirements();
-}
-
-void Return::checkTypeRequirements()
-{
- if (!m_expression)
- return;
- if (!m_returnParameters)
- BOOST_THROW_EXCEPTION(createTypeError("Return arguments not allowed."));
- if (m_returnParameters->parameters().size() != 1)
- BOOST_THROW_EXCEPTION(createTypeError(
- "Different number of arguments in return statement "
- "than in returns declaration."
- ));
- // this could later be changed such that the paramaters type is an anonymous struct type,
- // but for now, we only allow one return parameter
- m_expression->expectType(*m_returnParameters->parameters().front()->type());
-}
-
-void VariableDeclarationStatement::checkTypeRequirements()
-{
- m_variable->checkTypeRequirements();
-}
-
-void Assignment::checkTypeRequirements(TypePointers const*)
-{
- m_leftHandSide->checkTypeRequirements(nullptr);
- m_leftHandSide->requireLValue();
- if (m_leftHandSide->type()->category() == Type::Category::Mapping)
- BOOST_THROW_EXCEPTION(createTypeError("Mappings cannot be assigned to."));
- m_type = m_leftHandSide->type();
- if (m_assigmentOperator == Token::Assign)
- m_rightHandSide->expectType(*m_type);
- else
- {
- // compound assignment
- m_rightHandSide->checkTypeRequirements(nullptr);
- TypePointer resultType = m_type->binaryOperatorResult(Token::AssignmentToBinaryOp(m_assigmentOperator),
- m_rightHandSide->type());
- if (!resultType || *resultType != *m_type)
- BOOST_THROW_EXCEPTION(createTypeError(
- "Operator " +
- string(Token::toString(m_assigmentOperator)) +
- " not compatible with types " +
- m_type->toString() +
- " and " +
- m_rightHandSide->type()->toString()
- ));
- }
-}
-
-void ExpressionStatement::checkTypeRequirements()
-{
- m_expression->checkTypeRequirements(nullptr);
- if (m_expression->type()->category() == Type::Category::IntegerConstant)
- if (!dynamic_pointer_cast<IntegerConstantType const>(m_expression->type())->integerType())
- BOOST_THROW_EXCEPTION(m_expression->createTypeError("Invalid integer constant."));
-}
-
-void Expression::expectType(Type const& _expectedType)
-{
- checkTypeRequirements(nullptr);
- Type const& currentType = *type();
- if (!currentType.isImplicitlyConvertibleTo(_expectedType))
- BOOST_THROW_EXCEPTION(
- createTypeError(
- "Type " +
- currentType.toString() +
- " is not implicitly convertible to expected type " +
- _expectedType.toString() +
- "."
- )
- );
-}
-
-void Expression::requireLValue()
-{
- if (!isLValue())
- BOOST_THROW_EXCEPTION(createTypeError("Expression has to be an lvalue."));
- m_lvalueRequested = true;
-}
-
-void UnaryOperation::checkTypeRequirements(TypePointers const*)
-{
- // Inc, Dec, Add, Sub, Not, BitNot, Delete
- m_subExpression->checkTypeRequirements(nullptr);
- if (m_operator == Token::Value::Inc || m_operator == Token::Value::Dec || m_operator == Token::Value::Delete)
- m_subExpression->requireLValue();
- m_type = m_subExpression->type()->unaryOperatorResult(m_operator);
- if (!m_type)
- BOOST_THROW_EXCEPTION(createTypeError("Unary operator not compatible with type."));
-}
-
-void BinaryOperation::checkTypeRequirements(TypePointers const*)
-{
- m_left->checkTypeRequirements(nullptr);
- m_right->checkTypeRequirements(nullptr);
- m_commonType = m_left->type()->binaryOperatorResult(m_operator, m_right->type());
- if (!m_commonType)
- BOOST_THROW_EXCEPTION(createTypeError(
- "Operator " + string(Token::toString(m_operator)) +
- " not compatible with types " +
- m_left->type()->toString() +
- " and " +
- m_right->type()->toString()
- ));
- m_type = Token::isCompareOp(m_operator) ? make_shared<BoolType>() : m_commonType;
-}
-
-void FunctionCall::checkTypeRequirements(TypePointers const*)
-{
- bool isPositionalCall = m_names.empty();
-
- // we need to check arguments' type first as they will be forwarded to
- // m_expression->checkTypeRequirements
- TypePointers argumentTypes;
- for (ASTPointer<Expression> const& argument: m_arguments)
- {
- argument->checkTypeRequirements(nullptr);
- // only store them for positional calls
- if (isPositionalCall)
- argumentTypes.push_back(argument->type());
- }
-
- m_expression->checkTypeRequirements(isPositionalCall ? &argumentTypes : nullptr);
-
- TypePointer const& expressionType = m_expression->type();
- FunctionTypePointer functionType;
- if (isTypeConversion())
- {
- TypeType const& type = dynamic_cast<TypeType const&>(*expressionType);
- if (m_arguments.size() != 1)
- BOOST_THROW_EXCEPTION(createTypeError("Exactly one argument expected for explicit type conversion."));
- if (!isPositionalCall)
- BOOST_THROW_EXCEPTION(createTypeError("Type conversion cannot allow named arguments."));
- m_type = type.actualType();
- auto argType = m_arguments.front()->type();
- if (auto argRefType = dynamic_cast<ReferenceType const*>(argType.get()))
- // do not change the data location when converting
- // (data location cannot yet be specified for type conversions)
- m_type = ReferenceType::copyForLocationIfReference(argRefType->location(), m_type);
- if (!argType->isExplicitlyConvertibleTo(*m_type))
- BOOST_THROW_EXCEPTION(createTypeError("Explicit type conversion not allowed."));
-
- return;
- }
-
- /// For error message: Struct members that were removed during conversion to memory.
- set<string> membersRemovedForStructConstructor;
- if (isStructConstructorCall())
- {
- TypeType const& type = dynamic_cast<TypeType const&>(*expressionType);
- auto const& structType = dynamic_cast<StructType const&>(*type.actualType());
- functionType = structType.constructorType();
- membersRemovedForStructConstructor = structType.membersMissingInMemory();
- }
- else
- functionType = dynamic_pointer_cast<FunctionType const>(expressionType);
-
- if (!functionType)
- BOOST_THROW_EXCEPTION(createTypeError("Type is not callable."));
-
- //@todo would be nice to create a struct type from the arguments
- // and then ask if that is implicitly convertible to the struct represented by the
- // function parameters
- TypePointers const& parameterTypes = functionType->parameterTypes();
- if (!functionType->takesArbitraryParameters() && parameterTypes.size() != m_arguments.size())
- {
- string msg =
- "Wrong argument count for function call: " +
- toString(m_arguments.size()) +
- " arguments given but expected " +
- toString(parameterTypes.size()) +
- ".";
- // Extend error message in case we try to construct a struct with mapping member.
- if (isStructConstructorCall() && !membersRemovedForStructConstructor.empty())
- {
- msg += " Members that have to be skipped in memory:";
- for (auto const& member: membersRemovedForStructConstructor)
- msg += " " + member;
- }
- BOOST_THROW_EXCEPTION(createTypeError(msg));
- }
-
- if (isPositionalCall)
- {
- // call by positional arguments
- for (size_t i = 0; i < m_arguments.size(); ++i)
- if (
- !functionType->takesArbitraryParameters() &&
- !m_arguments[i]->type()->isImplicitlyConvertibleTo(*parameterTypes[i])
- )
- BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
- "Invalid type for argument in function call. "
- "Invalid implicit conversion from " +
- m_arguments[i]->type()->toString() +
- " to " +
- parameterTypes[i]->toString() +
- " requested."
- ));
- }
- else
- {
- // call by named arguments
- if (functionType->takesArbitraryParameters())
- BOOST_THROW_EXCEPTION(createTypeError(
- "Named arguments cannnot be used for functions that take arbitrary parameters."
- ));
- auto const& parameterNames = functionType->parameterNames();
- if (parameterNames.size() != m_names.size())
- BOOST_THROW_EXCEPTION(createTypeError("Some argument names are missing."));
-
- // check duplicate names
- for (size_t i = 0; i < m_names.size(); i++)
- for (size_t j = i + 1; j < m_names.size(); j++)
- if (*m_names[i] == *m_names[j])
- BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError("Duplicate named argument."));
-
- for (size_t i = 0; i < m_names.size(); i++) {
- bool found = false;
- for (size_t j = 0; j < parameterNames.size(); j++) {
- if (parameterNames[j] == *m_names[i]) {
- // check type convertible
- if (!m_arguments[i]->type()->isImplicitlyConvertibleTo(*parameterTypes[j]))
- BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
- "Invalid type for argument in function call. "
- "Invalid implicit conversion from " +
- m_arguments[i]->type()->toString() +
- " to " +
- parameterTypes[i]->toString() +
- " requested."
- ));
-
- found = true;
- break;
- }
- }
- if (!found)
- BOOST_THROW_EXCEPTION(createTypeError("Named argument does not match function declaration."));
- }
- }
-
- // @todo actually the return type should be an anonymous struct,
- // but we change it to the type of the first return value until we have anonymous
- // structs and tuples
- if (functionType->returnParameterTypes().empty())
- m_type = make_shared<VoidType>();
- else
- m_type = functionType->returnParameterTypes().front();
-}
-
-bool FunctionCall::isTypeConversion() const
-{
- return m_expression->type()->category() == Type::Category::TypeType && !isStructConstructorCall();
-}
-
-bool FunctionCall::isStructConstructorCall() const
-{
- if (auto const* type = dynamic_cast<TypeType const*>(m_expression->type().get()))
- return type->actualType()->category() == Type::Category::Struct;
- else
- return false;
-}
-
-void NewExpression::checkTypeRequirements(TypePointers const*)
-{
- m_contractName->checkTypeRequirements(nullptr);
- m_contract = dynamic_cast<ContractDefinition const*>(&m_contractName->referencedDeclaration());
-
- if (!m_contract)
- BOOST_THROW_EXCEPTION(createTypeError("Identifier is not a contract."));
- if (!m_contract->isFullyImplemented())
- BOOST_THROW_EXCEPTION(createTypeError("Trying to create an instance of an abstract contract."));
-
- auto scopeContract = m_contractName->contractScope();
- auto bases = m_contract->linearizedBaseContracts();
- if (find(bases.begin(), bases.end(), scopeContract) != bases.end())
- BOOST_THROW_EXCEPTION(createTypeError("Circular reference for contract creation: cannot create instance of derived or same contract."));
-
- shared_ptr<ContractType const> contractType = make_shared<ContractType>(*m_contract);
- TypePointers const& parameterTypes = contractType->constructorType()->parameterTypes();
- m_type = make_shared<FunctionType>(
- parameterTypes,
- TypePointers{contractType},
- strings(),
- strings(),
- FunctionType::Location::Creation
- );
-}
-
-void MemberAccess::checkTypeRequirements(TypePointers const* _argumentTypes)
-{
- m_expression->checkTypeRequirements(nullptr);
- Type const& type = *m_expression->type();
-
- MemberList::MemberMap possibleMembers = type.members().membersByName(*m_memberName);
- if (possibleMembers.size() > 1 && _argumentTypes)
- {
- // do override resolution
- for (auto it = possibleMembers.begin(); it != possibleMembers.end();)
- if (
- it->type->category() == Type::Category::Function &&
- !dynamic_cast<FunctionType const&>(*it->type).canTakeArguments(*_argumentTypes)
- )
- it = possibleMembers.erase(it);
- else
- ++it;
- }
- if (possibleMembers.size() == 0)
- {
- auto storageType = ReferenceType::copyForLocationIfReference(
- DataLocation::Storage,
- m_expression->type()
- );
- if (!storageType->members().membersByName(*m_memberName).empty())
- BOOST_THROW_EXCEPTION(createTypeError(
- "Member \"" +
- *m_memberName +
- "\" is not available in " +
- type.toString() +
- " outside of storage."
- ));
- BOOST_THROW_EXCEPTION(createTypeError(
- "Member \"" +
- *m_memberName +
- "\" not found or not visible after argument-dependent lookup in " +
- type.toString()
- ));
- }
- else if (possibleMembers.size() > 1)
- BOOST_THROW_EXCEPTION(createTypeError(
- "Member \"" +
- *m_memberName +
- "\" not unique after argument-dependent lookup in " +
- type.toString()
- ));
-
- m_referencedDeclaration = possibleMembers.front().declaration;
- m_type = possibleMembers.front().type;
- if (type.category() == Type::Category::Struct)
- m_isLValue = true;
- else if (type.category() == Type::Category::Array)
- {
- auto const& arrayType(dynamic_cast<ArrayType const&>(type));
- m_isLValue = (
- *m_memberName == "length" &&
- arrayType.location() == DataLocation::Storage &&
- arrayType.isDynamicallySized()
- );
- }
- else
- m_isLValue = false;
-}
-
-void IndexAccess::checkTypeRequirements(TypePointers const*)
-{
- m_base->checkTypeRequirements(nullptr);
- switch (m_base->type()->category())
- {
- case Type::Category::Array:
- {
- ArrayType const& type = dynamic_cast<ArrayType const&>(*m_base->type());
- if (!m_index)
- BOOST_THROW_EXCEPTION(createTypeError("Index expression cannot be omitted."));
- if (type.isString())
- BOOST_THROW_EXCEPTION(createTypeError("Index access for string is not possible."));
- m_index->expectType(IntegerType(256));
-
- m_type = type.baseType();
- if (auto integerType = dynamic_cast<IntegerConstantType const*>(m_index->type().get()))
- if (!type.isDynamicallySized() && type.length() <= integerType->literalValue(nullptr))
- BOOST_THROW_EXCEPTION(createTypeError("Out of bounds access."));
-
- m_isLValue = type.location() != DataLocation::CallData;
- break;
- }
- case Type::Category::Mapping:
- {
- MappingType const& type = dynamic_cast<MappingType const&>(*m_base->type());
- if (!m_index)
- BOOST_THROW_EXCEPTION(createTypeError("Index expression cannot be omitted."));
- m_index->expectType(*type.keyType());
- m_type = type.valueType();
- m_isLValue = true;
- break;
- }
- case Type::Category::TypeType:
- {
- TypeType const& type = dynamic_cast<TypeType const&>(*m_base->type());
- if (!m_index)
- m_type = make_shared<TypeType>(make_shared<ArrayType>(DataLocation::Memory, type.actualType()));
- else
- {
- m_index->checkTypeRequirements(nullptr);
- auto length = dynamic_cast<IntegerConstantType const*>(m_index->type().get());
- if (!length)
- BOOST_THROW_EXCEPTION(m_index->createTypeError("Integer constant expected."));
- m_type = make_shared<TypeType>(make_shared<ArrayType>(
- DataLocation::Memory,
- type.actualType(),
- length->literalValue(nullptr)
- ));
- }
- break;
- }
- default:
- BOOST_THROW_EXCEPTION(m_base->createTypeError(
- "Indexed expression has to be a type, mapping or array (is " +
- m_base->type()->toString() +
- ")"
- ));
- }
-}
-
-void Identifier::checkTypeRequirements(TypePointers const* _argumentTypes)
-{
- if (!m_referencedDeclaration)
- {
- if (!_argumentTypes)
- BOOST_THROW_EXCEPTION(createTypeError("Unable to determine overloaded type."));
- overloadResolution(*_argumentTypes);
- }
- solAssert(!!m_referencedDeclaration, "Referenced declaration is null after overload resolution.");
- m_isLValue = m_referencedDeclaration->isLValue();
- m_type = m_referencedDeclaration->type(m_contractScope);
- if (!m_type)
- BOOST_THROW_EXCEPTION(createTypeError("Declaration referenced before type could be determined."));
-}
-
-Declaration const& Identifier::referencedDeclaration() const
-{
- solAssert(!!m_referencedDeclaration, "Identifier not resolved.");
- return *m_referencedDeclaration;
-}
-
-void Identifier::overloadResolution(TypePointers const& _argumentTypes)
-{
- solAssert(!m_referencedDeclaration, "Referenced declaration should be null before overload resolution.");
- solAssert(!m_overloadedDeclarations.empty(), "No candidates for overload resolution found.");
-
- vector<Declaration const*> possibles;
- if (m_overloadedDeclarations.size() == 1)
- m_referencedDeclaration = *m_overloadedDeclarations.begin();
-
- for (Declaration const* declaration: m_overloadedDeclarations)
- {
- TypePointer const& function = declaration->type();
- auto const* functionType = dynamic_cast<FunctionType const*>(function.get());
- if (functionType && functionType->canTakeArguments(_argumentTypes))
- possibles.push_back(declaration);
- }
- if (possibles.size() == 1)
- m_referencedDeclaration = possibles.front();
- else if (possibles.empty())
- BOOST_THROW_EXCEPTION(createTypeError("No matching declaration found after argument-dependent lookup."));
- else
- BOOST_THROW_EXCEPTION(createTypeError("No unique declaration found after argument-dependent lookup."));
-}
-
-void ElementaryTypeNameExpression::checkTypeRequirements(TypePointers const*)
-{
- m_type = make_shared<TypeType>(Type::fromElementaryTypeName(m_typeToken));
+ auto const* callable = dynamic_cast<CallableDeclaration const*>(scope());
+ return (!!callable && !isCallableParameter());
}
-void Literal::checkTypeRequirements(TypePointers const*)
+TypePointer VariableDeclaration::type(ContractDefinition const*) const
{
- m_type = Type::forLiteral(*this);
- if (!m_type)
- BOOST_THROW_EXCEPTION(createTypeError("Invalid literal value."));
-}
-
-}
+ return annotation().type;
}