solidity/libsolidity/AST.cpp

1235 lines
41 KiB
C++
Raw Normal View History

/*
2014-10-16 12:08:54 +00:00
This file is part of cpp-ethereum.
2014-10-16 12:08:54 +00:00
cpp-ethereum is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
2014-10-16 12:08:54 +00:00
cpp-ethereum is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2014-10-16 12:08:54 +00:00
You should have received a copy of the GNU General Public License
along with cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* @author Christian <c@ethdev.com>
* @date 2014
* Solidity abstract syntax tree.
*/
2014-10-13 16:22:15 +00:00
#include <algorithm>
#include <functional>
#include <boost/range/adaptor/reversed.hpp>
#include <libsolidity/Utils.h>
#include <libsolidity/AST.h>
#include <libsolidity/ASTVisitor.h>
2014-10-15 12:45:51 +00:00
#include <libsolidity/Exceptions.h>
#include <libsolidity/AST_accept.h>
#include <libdevcore/SHA3.h>
2014-10-24 17:06:30 +00:00
using namespace std;
2014-10-16 12:08:54 +00:00
namespace dev
{
namespace solidity
{
TypeError ASTNode::createTypeError(string const& _description) const
2014-10-23 17:22:30 +00:00
{
2015-08-31 16:44:29 +00:00
return TypeError() << errinfo_sourceLocation(location()) << errinfo_comment(_description);
2014-10-23 17:22:30 +00:00
}
2015-08-31 16:44:29 +00:00
TypePointer ContractDefinition::type(ContractDefinition const* _currentContract) const
{
return make_shared<TypeType>(make_shared<ContractType>(*this), _currentContract);
}
void ContractDefinition::checkTypeRequirements()
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<InheritanceSpecifier> const& baseSpecifier: baseContracts())
baseSpecifier->checkTypeRequirements();
checkDuplicateFunctions();
checkIllegalOverrides();
checkAbstractFunctions();
checkAbstractConstructors();
FunctionDefinition const* functionDefinition = constructor();
2015-09-09 16:18:34 +00:00
if (functionDefinition && !functionDefinition->returnParameters().empty())
BOOST_THROW_EXCEPTION(functionDefinition->returnParameterList()->createTypeError(
"Non-empty \"returns\" directive for constructor."
));
2015-01-29 21:50:20 +00:00
FunctionDefinition const* fallbackFunction = nullptr;
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
{
2015-08-31 16:44:29 +00:00
if (function->name().empty())
{
if (fallbackFunction)
BOOST_THROW_EXCEPTION(DeclarationError() << errinfo_comment("Only one fallback function is allowed."));
else
{
fallbackFunction = function.get();
2015-08-31 16:44:29 +00:00
if (!fallbackFunction->parameters().empty())
BOOST_THROW_EXCEPTION(fallbackFunction->parameterList().createTypeError("Fallback function cannot take parameters."));
}
}
if (!function->isFullyImplemented())
setFullyImplemented(false);
}
2015-09-08 10:57:27 +00:00
for (ASTPointer<VariableDeclaration> const& variable: m_stateVariables)
variable->checkTypeRequirements();
for (ASTPointer<EventDefinition> const& event: events())
event->checkTypeRequirements();
2015-08-31 16:44:29 +00:00
for (ASTPointer<ModifierDefinition> const& modifier: functionModifiers())
modifier->checkTypeRequirements();
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
function->checkTypeRequirements();
checkExternalTypeClashes();
// check for hash collisions in function signatures
set<FixedHash<4>> hashes;
2015-08-31 16:44:29 +00:00
for (auto const& it: interfaceFunctionList())
{
FixedHash<4> const& hash = it.first;
if (hashes.count(hash))
2015-05-04 15:28:28 +00:00
BOOST_THROW_EXCEPTION(createTypeError(
string("Function signature hash collision for ") + it.second->externalSignature()
));
hashes.insert(hash);
}
if (isLibrary())
checkLibraryRequirements();
}
2015-08-31 16:44:29 +00:00
map<FixedHash<4>, FunctionTypePointer> ContractDefinition::interfaceFunctions() const
{
2015-08-31 16:44:29 +00:00
auto exportedFunctionList = interfaceFunctionList();
map<FixedHash<4>, FunctionTypePointer> exportedFunctions;
for (auto const& it: exportedFunctionList)
exportedFunctions.insert(it);
2015-09-10 08:17:17 +00:00
solAssert(
exportedFunctionList.size() == exportedFunctions.size(),
"Hash collision at Function Definition Hash calculation"
);
return exportedFunctions;
}
2015-08-31 16:44:29 +00:00
FunctionDefinition const* ContractDefinition::constructor() const
2014-12-12 15:49:26 +00:00
{
for (ASTPointer<FunctionDefinition> const& f: m_definedFunctions)
if (f->isConstructor())
2014-12-12 15:49:26 +00:00
return f.get();
return nullptr;
}
2015-08-31 16:44:29 +00:00
FunctionDefinition const* ContractDefinition::fallbackFunction() const
2015-01-29 21:50:20 +00:00
{
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: linearizedBaseContracts())
for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
if (f->name().empty())
2015-01-29 21:50:20 +00:00
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;
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& function: definedFunctions())
functions[function->name()].push_back(function.get());
2015-08-31 16:44:29 +00:00
if (functions[name()].size() > 1)
{
SecondarySourceLocation ssl;
2015-08-31 16:44:29 +00:00
auto it = functions[name()].begin();
++it;
2015-08-31 16:44:29 +00:00
for (; it != functions[name()].end(); ++it)
ssl.append("Another declaration is here:", (*it)->location());
BOOST_THROW_EXCEPTION(
DeclarationError() <<
2015-08-31 16:44:29 +00:00
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() <<
2015-08-31 16:44:29 +00:00
errinfo_sourceLocation(overloads[j]->location()) <<
2015-05-04 13:46:46 +00:00
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
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: boost::adaptors::reverse(linearizedBaseContracts()))
for (ASTPointer<FunctionDefinition> const& function: contract->definedFunctions())
{
2015-08-31 16:44:29 +00:00
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()
{
2015-04-17 14:17:21 +00:00
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)
2015-04-17 14:17:21 +00:00
2015-08-31 16:44:29 +00:00
vector<ContractDefinition const*> const& bases = linearizedBaseContracts();
2015-04-17 14:17:21 +00:00
for (ContractDefinition const* contract: bases)
2015-08-31 16:44:29 +00:00
if (FunctionDefinition const* constructor = contract->constructor())
if (contract != this && !constructor->parameters().empty())
2015-04-17 14:17:21 +00:00
argumentsNeeded.insert(contract);
for (ContractDefinition const* contract: bases)
{
2015-08-31 16:44:29 +00:00
if (FunctionDefinition const* constructor = contract->constructor())
for (auto const& modifier: constructor->modifiers())
{
auto baseContract = dynamic_cast<ContractDefinition const*>(
2015-08-31 16:44:29 +00:00
&modifier->name()->referencedDeclaration()
2015-04-17 14:17:21 +00:00
);
if (baseContract)
2015-04-17 14:17:21 +00:00
argumentsNeeded.erase(baseContract);
}
2015-04-17 14:17:21 +00:00
2015-08-31 16:44:29 +00:00
for (ASTPointer<InheritanceSpecifier> const& base: contract->baseContracts())
{
2015-04-17 14:17:21 +00:00
auto baseContract = dynamic_cast<ContractDefinition const*>(
2015-08-31 16:44:29 +00:00
&base->name()->referencedDeclaration()
2015-04-17 14:17:21 +00:00
);
solAssert(baseContract, "");
2015-08-31 16:44:29 +00:00
if (!base->arguments().empty())
2015-04-17 14:17:21 +00:00
argumentsNeeded.erase(baseContract);
}
}
2015-04-17 14:17:21 +00:00
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.
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: linearizedBaseContracts())
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& function: contract->definedFunctions())
{
if (function->isConstructor())
continue; // constructors can neither be overridden nor override anything
2015-08-31 16:44:29 +00:00
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 (
2015-08-31 16:44:29 +00:00
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());
}
2015-08-31 16:44:29 +00:00
for (ASTPointer<ModifierDefinition> const& modifier: contract->functionModifiers())
{
2015-08-31 16:44:29 +00:00
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())
2015-03-01 03:34:39 +00:00
BOOST_THROW_EXCEPTION(override->createTypeError("Override changes modifier to function."));
}
}
}
void ContractDefinition::checkExternalTypeClashes() const
{
map<string, vector<pair<Declaration const*, shared_ptr<FunctionType>>>> externalDeclarations;
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: linearizedBaseContracts())
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
if (f->isPartOfExternalInterface())
{
auto functionType = make_shared<FunctionType>(*f);
2015-08-31 16:44:29 +00:00
externalDeclarations[functionType->externalSignature(f->name())].push_back(
make_pair(f.get(), functionType)
);
}
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& v: contract->stateVariables())
if (v->isPartOfExternalInterface())
{
auto functionType = make_shared<FunctionType>(*v);
2015-08-31 16:44:29 +00:00
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"));
}
2015-08-31 16:44:29 +00:00
vector<ASTPointer<EventDefinition>> const& ContractDefinition::interfaceEvents() const
{
if (!m_interfaceEvents)
{
set<string> eventsSeen;
2015-05-04 15:18:01 +00:00
m_interfaceEvents.reset(new vector<ASTPointer<EventDefinition>>());
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: linearizedBaseContracts())
for (ASTPointer<EventDefinition> const& e: contract->events())
if (eventsSeen.count(e->name()) == 0)
{
2015-08-31 16:44:29 +00:00
eventsSeen.insert(e->name());
m_interfaceEvents->push_back(e);
}
}
return *m_interfaceEvents;
}
2015-08-31 16:44:29 +00:00
vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::interfaceFunctionList() const
{
if (!m_interfaceFunctionList)
{
set<string> functionsSeen;
2015-03-01 03:34:39 +00:00
set<string> signaturesSeen;
2015-01-29 16:28:14 +00:00
m_interfaceFunctionList.reset(new vector<pair<FixedHash<4>, FunctionTypePointer>>());
2015-08-31 16:44:29 +00:00
for (ContractDefinition const* contract: linearizedBaseContracts())
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& f: contract->definedFunctions())
2015-03-01 03:34:39 +00:00
{
if (!f->isPartOfExternalInterface())
continue;
string functionSignature = f->externalSignature();
if (signaturesSeen.count(functionSignature) == 0)
{
2015-08-31 16:44:29 +00:00
functionsSeen.insert(f->name());
2015-03-01 03:34:39 +00:00
signaturesSeen.insert(functionSignature);
FixedHash<4> hash(dev::sha3(functionSignature));
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*f, false)));
}
2015-03-01 03:34:39 +00:00
}
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& v: contract->stateVariables())
if (functionsSeen.count(v->name()) == 0 && v->isPartOfExternalInterface())
{
FunctionType ftype(*v);
2015-08-31 16:44:29 +00:00
solAssert(v->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)));
}
}
}
return *m_interfaceFunctionList;
}
string const& ContractDefinition::devDocumentation() const
{
return m_devDocumentation;
}
string const& ContractDefinition::userDocumentation() const
{
return m_userDocumentation;
}
void ContractDefinition::setDevDocumentation(string const& _devDocumentation)
{
m_devDocumentation = _devDocumentation;
}
void ContractDefinition::setUserDocumentation(string const& _userDocumentation)
{
m_userDocumentation = _userDocumentation;
}
2015-08-31 16:44:29 +00:00
vector<Declaration const*> const& ContractDefinition::inheritableMembers() const
{
if (!m_inheritableMembers)
{
set<string> memberSeen;
m_inheritableMembers.reset(new vector<Declaration const*>());
auto addInheritableMember = [&](Declaration const* _decl)
{
2015-08-31 16:44:29 +00:00
if (memberSeen.count(_decl->name()) == 0 && _decl->isVisibleInDerivedContracts())
{
2015-08-31 16:44:29 +00:00
memberSeen.insert(_decl->name());
m_inheritableMembers->push_back(_decl);
}
};
2015-08-31 16:44:29 +00:00
for (ASTPointer<FunctionDefinition> const& f: definedFunctions())
addInheritableMember(f.get());
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& v: stateVariables())
addInheritableMember(v.get());
2015-02-26 11:11:54 +00:00
2015-08-31 16:44:29 +00:00
for (ASTPointer<StructDefinition> const& s: definedStructs())
addInheritableMember(s.get());
}
return *m_inheritableMembers;
}
2015-08-31 16:44:29 +00:00
TypePointer EnumValue::type(ContractDefinition const*) const
2015-02-10 12:40:21 +00:00
{
2015-08-31 16:44:29 +00:00
EnumDefinition const* parentDef = dynamic_cast<EnumDefinition const*>(scope());
solAssert(parentDef, "Enclosing Scope of EnumValue was not set");
2015-02-13 16:32:34 +00:00
return make_shared<EnumType>(*parentDef);
2015-02-10 12:40:21 +00:00
}
void InheritanceSpecifier::checkTypeRequirements()
{
m_baseName->checkTypeRequirements(nullptr);
for (ASTPointer<Expression> const& argument: m_arguments)
argument->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
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."));
2015-08-31 16:44:29 +00:00
TypePointers parameterTypes = ContractType(*base).constructorType()->parameterTypes();
2015-04-02 15:05:27 +00:00
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)
2015-08-31 16:44:29 +00:00
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 " +
2015-08-31 16:44:29 +00:00
m_arguments[i]->type()->toString() +
" to " +
parameterTypes[i]->toString() +
" requested."
));
}
2015-08-31 16:44:29 +00:00
TypePointer StructDefinition::type(ContractDefinition const*) const
{
return make_shared<TypeType>(make_shared<StructType>(*this));
}
void StructDefinition::checkMemberTypes() const
2014-12-03 06:46:55 +00:00
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& member: members())
if (!member->type()->canBeStored())
2014-12-03 06:46:55 +00:00
BOOST_THROW_EXCEPTION(member->createTypeError("Type cannot be used in struct."));
}
void StructDefinition::checkRecursion() const
2014-12-03 06:46:55 +00:00
{
using StructPointer = StructDefinition const*;
using StructPointersSet = set<StructPointer>;
function<void(StructPointer,StructPointersSet const&)> check = [&](StructPointer _struct, StructPointersSet const& _parents)
2014-12-03 06:46:55 +00:00
{
if (_parents.count(_struct))
BOOST_THROW_EXCEPTION(
ParserError() <<
2015-08-31 16:44:29 +00:00
errinfo_sourceLocation(_struct->location()) <<
errinfo_comment("Recursive struct definition.")
);
set<StructDefinition const*> parents = _parents;
parents.insert(_struct);
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& member: _struct->members())
if (member->type()->category() == Type::Category::Struct)
2014-12-03 06:46:55 +00:00
{
2015-08-31 16:44:29 +00:00
auto const& typeName = dynamic_cast<UserDefinedTypeName const&>(*member->typeName());
check(
2015-08-31 16:44:29 +00:00
&dynamic_cast<StructDefinition const&>(*typeName.referencedDeclaration()),
parents
);
2014-12-03 06:46:55 +00:00
}
};
check(this, StructPointersSet{});
2014-12-03 06:46:55 +00:00
}
2015-08-31 16:44:29 +00:00
TypePointer EnumDefinition::type(ContractDefinition const*) const
2015-02-09 17:08:56 +00:00
{
return make_shared<TypeType>(make_shared<EnumType>(*this));
2015-02-09 17:08:56 +00:00
}
2015-08-31 16:44:29 +00:00
TypePointer FunctionDefinition::type(ContractDefinition const*) const
{
return make_shared<FunctionType>(*this);
}
2014-11-10 16:31:09 +00:00
void FunctionDefinition::checkTypeRequirements()
{
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& var: parameters() + returnParameters())
{
2015-08-31 16:44:29 +00:00
if (!var->type()->canLiveOutsideStorage())
2014-11-10 16:31:09 +00:00
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
2015-08-31 16:44:29 +00:00
if (visibility() >= Visibility::Public && !(var->type()->externalType()))
2015-04-21 08:59:48 +00:00
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() ?
2015-08-31 16:44:29 +00:00
dynamic_cast<ContractDefinition const&>(*scope()).linearizedBaseContracts() :
vector<ContractDefinition const*>());
if (m_body)
m_body->checkTypeRequirements();
2014-11-10 16:31:09 +00:00
}
string FunctionDefinition::externalSignature() const
{
2015-08-31 16:44:29 +00:00
return FunctionType(*this).externalSignature(name());
}
bool VariableDeclaration::isLValue() const
{
2015-03-03 11:58:01 +00:00
// External function parameters and constant declared variables are Read-Only
return !isExternalCallableParameter() && !m_isConstant;
}
void VariableDeclaration::checkTypeRequirements()
{
// 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)
{
2015-08-31 16:44:29 +00:00
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."));
2015-09-08 10:57:27 +00:00
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\" "
2015-09-09 14:15:35 +00:00
"is not yet implemented for this type."
2015-09-08 10:57:27 +00:00
));
}
}
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);
2015-03-08 22:26:36 +00:00
2015-08-31 16:44:29 +00:00
TypePointer const& type = m_value->type();
if (
2015-08-31 16:44:29 +00:00
type->category() == Type::Category::IntegerConstant &&
!dynamic_pointer_cast<IntegerConstantType const>(type)->integerType()
)
BOOST_THROW_EXCEPTION(m_value->createTypeError("Invalid integer constant " + type->toString() + "."));
2015-08-31 16:44:29 +00:00
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."
));
}
2015-08-31 16:44:29 +00:00
else if (visibility() >= Visibility::Public && !FunctionType(*this).externalType())
BOOST_THROW_EXCEPTION(createTypeError("Internal type is not allowed for public state variables."));
}
bool VariableDeclaration::isCallableParameter() const
{
2015-08-31 16:44:29 +00:00
auto const* callable = dynamic_cast<CallableDeclaration const*>(scope());
if (!callable)
return false;
2015-08-31 16:44:29 +00:00
for (auto const& variable: callable->parameters())
if (variable.get() == this)
return true;
2015-08-31 16:44:29 +00:00
if (callable->returnParameterList())
for (auto const& variable: callable->returnParameterList()->parameters())
if (variable.get() == this)
return true;
return false;
}
bool VariableDeclaration::isExternalCallableParameter() const
{
2015-08-31 16:44:29 +00:00
auto const* callable = dynamic_cast<CallableDeclaration const*>(scope());
if (!callable || callable->visibility() != Declaration::Visibility::External)
return false;
2015-08-31 16:44:29 +00:00
for (auto const& variable: callable->parameters())
if (variable.get() == this)
return true;
return false;
}
2015-08-31 16:44:29 +00:00
TypePointer ModifierDefinition::type(ContractDefinition const*) const
{
return make_shared<ModifierType>(*this);
}
2015-01-21 10:16:18 +00:00
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);
2015-08-31 16:44:29 +00:00
argumentTypes.push_back(argument->type());
}
m_modifierName->checkTypeRequirements(&argumentTypes);
2015-08-31 16:44:29 +00:00
auto const* declaration = &m_modifierName->referencedDeclaration();
vector<ASTPointer<VariableDeclaration>> emptyParameterList;
vector<ASTPointer<VariableDeclaration>> const* parameters = nullptr;
if (auto modifier = dynamic_cast<ModifierDefinition const*>(declaration))
2015-08-31 16:44:29 +00:00
parameters = &modifier->parameters();
else
// check parameters for Base constructors
for (ContractDefinition const* base: _bases)
if (declaration == base)
{
2015-08-31 16:44:29 +00:00
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)
2015-08-31 16:44:29 +00:00
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 " +
2015-08-31 16:44:29 +00:00
m_arguments[i]->type()->toString() +
" to " +
2015-08-31 16:44:29 +00:00
(*parameters)[i]->type()->toString() +
" requested."
));
}
2015-01-29 13:35:28 +00:00
void EventDefinition::checkTypeRequirements()
{
int numIndexed = 0;
2015-08-31 16:44:29 +00:00
for (ASTPointer<VariableDeclaration> const& var: parameters())
2015-01-29 13:35:28 +00:00
{
if (var->isIndexed())
numIndexed++;
if (numIndexed > 3)
BOOST_THROW_EXCEPTION(createTypeError("More than 3 indexed arguments for event."));
2015-08-31 16:44:29 +00:00
if (!var->type()->canLiveOutsideStorage())
2015-01-29 13:35:28 +00:00
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
2015-08-31 16:44:29 +00:00
if (!var->type()->externalType())
2015-03-26 13:11:24 +00:00
BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed as event parameter type."));
2015-01-29 13:35:28 +00:00
}
}
void Block::checkTypeRequirements()
2014-10-13 16:22:15 +00:00
{
2014-10-24 17:06:30 +00:00
for (shared_ptr<Statement> const& statement: m_statements)
2014-10-13 16:22:15 +00:00
statement->checkTypeRequirements();
}
void IfStatement::checkTypeRequirements()
2014-10-13 16:22:15 +00:00
{
m_condition->expectType(BoolType());
2014-10-13 16:22:15 +00:00
m_trueBody->checkTypeRequirements();
2014-10-16 21:49:45 +00:00
if (m_falseBody)
m_falseBody->checkTypeRequirements();
2014-10-13 16:22:15 +00:00
}
void WhileStatement::checkTypeRequirements()
2014-10-13 16:22:15 +00:00
{
m_condition->expectType(BoolType());
2014-10-13 16:22:15 +00:00
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()
2014-10-13 16:22:15 +00:00
{
if (!m_expression)
return;
2015-01-23 01:35:27 +00:00
if (!m_returnParameters)
BOOST_THROW_EXCEPTION(createTypeError("Return arguments not allowed."));
2015-08-31 16:44:29 +00:00
if (m_returnParameters->parameters().size() != 1)
BOOST_THROW_EXCEPTION(createTypeError(
"Different number of arguments in return statement "
"than in returns declaration."
));
2014-10-13 16:22:15 +00:00
// this could later be changed such that the paramaters type is an anonymous struct type,
// but for now, we only allow one return parameter
2015-08-31 16:44:29 +00:00
m_expression->expectType(*m_returnParameters->parameters().front()->type());
2014-10-13 16:22:15 +00:00
}
void VariableDeclarationStatement::checkTypeRequirements()
2014-10-13 16:22:15 +00:00
{
m_variable->checkTypeRequirements();
2014-10-13 16:22:15 +00:00
}
void Assignment::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
m_leftHandSide->checkTypeRequirements(nullptr);
2014-11-10 16:31:09 +00:00
m_leftHandSide->requireLValue();
2015-08-31 16:44:29 +00:00
if (m_leftHandSide->type()->category() == Type::Category::Mapping)
2015-02-12 17:38:07 +00:00
BOOST_THROW_EXCEPTION(createTypeError("Mappings cannot be assigned to."));
2015-08-31 16:44:29 +00:00
m_type = m_leftHandSide->type();
if (m_assigmentOperator == Token::Assign)
m_rightHandSide->expectType(*m_type);
else
{
2014-10-20 10:41:56 +00:00
// compound assignment
m_rightHandSide->checkTypeRequirements(nullptr);
TypePointer resultType = m_type->binaryOperatorResult(Token::AssignmentToBinaryOp(m_assigmentOperator),
2015-08-31 16:44:29 +00:00
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()
));
}
2014-10-13 16:22:15 +00:00
}
void ExpressionStatement::checkTypeRequirements()
{
m_expression->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
if (m_expression->type()->category() == Type::Category::IntegerConstant)
if (!dynamic_pointer_cast<IntegerConstantType const>(m_expression->type())->integerType())
2014-12-19 10:31:17 +00:00
BOOST_THROW_EXCEPTION(m_expression->createTypeError("Invalid integer constant."));
}
2014-11-05 14:04:33 +00:00
void Expression::expectType(Type const& _expectedType)
{
checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
Type const& currentType = *type();
if (!currentType.isImplicitlyConvertibleTo(_expectedType))
BOOST_THROW_EXCEPTION(
createTypeError(
"Type " +
currentType.toString() +
" is not implicitly convertible to expected type " +
_expectedType.toString() +
"."
)
);
}
2014-11-10 16:31:09 +00:00
void Expression::requireLValue()
{
if (!isLValue())
2014-11-10 16:31:09 +00:00
BOOST_THROW_EXCEPTION(createTypeError("Expression has to be an lvalue."));
m_lvalueRequested = true;
}
void UnaryOperation::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
2015-02-10 08:52:19 +00:00
// 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)
2014-11-10 16:31:09 +00:00
m_subExpression->requireLValue();
2015-08-31 16:44:29 +00:00
m_type = m_subExpression->type()->unaryOperatorResult(m_operator);
if (!m_type)
2014-10-23 17:22:30 +00:00
BOOST_THROW_EXCEPTION(createTypeError("Unary operator not compatible with type."));
2014-10-13 16:22:15 +00:00
}
void BinaryOperation::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
m_left->checkTypeRequirements(nullptr);
m_right->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
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;
2014-10-13 16:22:15 +00:00
}
void FunctionCall::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
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)
2015-08-31 16:44:29 +00:00
argumentTypes.push_back(argument->type());
}
2014-10-20 10:41:56 +00:00
m_expression->checkTypeRequirements(isPositionalCall ? &argumentTypes : nullptr);
2015-03-01 03:34:39 +00:00
2015-08-31 16:44:29 +00:00
TypePointer const& expressionType = m_expression->type();
2015-06-30 19:08:34 +00:00
FunctionTypePointer functionType;
2014-10-20 10:41:56 +00:00
if (isTypeConversion())
2014-10-16 12:08:54 +00:00
{
2014-11-05 13:20:56 +00:00
TypeType const& type = dynamic_cast<TypeType const&>(*expressionType);
2014-10-13 16:22:15 +00:00
if (m_arguments.size() != 1)
BOOST_THROW_EXCEPTION(createTypeError("Exactly one argument expected for explicit type conversion."));
if (!isPositionalCall)
2015-02-08 23:49:35 +00:00
BOOST_THROW_EXCEPTION(createTypeError("Type conversion cannot allow named arguments."));
2015-08-31 16:44:29 +00:00
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."));
2015-06-30 19:08:34 +00:00
return;
2014-10-16 12:08:54 +00:00
}
2015-06-30 19:08:34 +00:00
/// For error message: Struct members that were removed during conversion to memory.
set<string> membersRemovedForStructConstructor;
2015-06-30 19:08:34 +00:00
if (isStructConstructorCall())
2014-10-16 12:08:54 +00:00
{
2015-06-30 19:08:34 +00:00
TypeType const& type = dynamic_cast<TypeType const&>(*expressionType);
2015-08-31 16:44:29 +00:00
auto const& structType = dynamic_cast<StructType const&>(*type.actualType());
2015-06-30 19:08:34 +00:00
functionType = structType.constructorType();
membersRemovedForStructConstructor = structType.membersMissingInMemory();
2015-06-30 19:08:34 +00:00
}
else
functionType = dynamic_pointer_cast<FunctionType const>(expressionType);
2015-01-29 17:26:00 +00:00
2015-06-30 19:08:34 +00:00
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
2015-08-31 16:44:29 +00:00
TypePointers const& parameterTypes = functionType->parameterTypes();
2015-06-30 19:08:34 +00:00
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));
}
2015-06-30 19:08:34 +00:00
if (isPositionalCall)
{
// call by positional arguments
for (size_t i = 0; i < m_arguments.size(); ++i)
if (
!functionType->takesArbitraryParameters() &&
2015-08-31 16:44:29 +00:00
!m_arguments[i]->type()->isImplicitlyConvertibleTo(*parameterTypes[i])
2015-06-30 19:08:34 +00:00
)
BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
"Invalid type for argument in function call. "
"Invalid implicit conversion from " +
2015-08-31 16:44:29 +00:00
m_arguments[i]->type()->toString() +
2015-06-30 19:08:34 +00:00
" 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."
));
2015-08-31 16:44:29 +00:00
auto const& parameterNames = functionType->parameterNames();
2015-06-30 19:08:34 +00:00
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
2015-08-31 16:44:29 +00:00
if (!m_arguments[i]->type()->isImplicitlyConvertibleTo(*parameterTypes[j]))
2015-06-30 19:08:34 +00:00
BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError(
"Invalid type for argument in function call. "
"Invalid implicit conversion from " +
2015-08-31 16:44:29 +00:00
m_arguments[i]->type()->toString() +
2015-06-30 19:08:34 +00:00
" to " +
parameterTypes[i]->toString() +
" requested."
));
found = true;
break;
2015-01-29 17:26:00 +00:00
}
}
2015-06-30 19:08:34 +00:00
if (!found)
BOOST_THROW_EXCEPTION(createTypeError("Named argument does not match function declaration."));
2015-01-29 17:26:00 +00:00
}
2014-10-16 12:08:54 +00:00
}
2015-06-30 19:08:34 +00:00
// @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
2015-08-31 16:44:29 +00:00
if (functionType->returnParameterTypes().empty())
2015-06-30 19:08:34 +00:00
m_type = make_shared<VoidType>();
2014-12-12 15:49:26 +00:00
else
2015-08-31 16:44:29 +00:00
m_type = functionType->returnParameterTypes().front();
2014-10-20 10:41:56 +00:00
}
bool FunctionCall::isTypeConversion() const
{
2015-08-31 16:44:29 +00:00
return m_expression->type()->category() == Type::Category::TypeType && !isStructConstructorCall();
2015-06-30 19:08:34 +00:00
}
bool FunctionCall::isStructConstructorCall() const
{
2015-08-31 16:44:29 +00:00
if (auto const* type = dynamic_cast<TypeType const*>(m_expression->type().get()))
return type->actualType()->category() == Type::Category::Struct;
2015-06-30 19:08:34 +00:00
else
return false;
2014-10-13 16:22:15 +00:00
}
void NewExpression::checkTypeRequirements(TypePointers const*)
2014-12-12 15:49:26 +00:00
{
m_contractName->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
m_contract = dynamic_cast<ContractDefinition const*>(&m_contractName->referencedDeclaration());
2014-12-12 15:49:26 +00:00
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."));
2015-01-13 17:12:19 +00:00
shared_ptr<ContractType const> contractType = make_shared<ContractType>(*m_contract);
2015-08-31 16:44:29 +00:00
TypePointers const& parameterTypes = contractType->constructorType()->parameterTypes();
m_type = make_shared<FunctionType>(
parameterTypes,
TypePointers{contractType},
strings(),
strings(),
FunctionType::Location::Creation
);
2014-12-12 15:49:26 +00:00
}
void MemberAccess::checkTypeRequirements(TypePointers const* _argumentTypes)
2014-10-13 16:22:15 +00:00
{
m_expression->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
Type const& type = *m_expression->type();
2015-08-31 16:44:29 +00:00
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 (
2015-08-31 16:44:29 +00:00
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,
2015-08-31 16:44:29 +00:00
m_expression->type()
);
2015-08-31 16:44:29 +00:00
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;
2015-08-31 16:44:29 +00:00
if (type.category() == Type::Category::Struct)
m_isLValue = true;
2015-08-31 16:44:29 +00:00
else if (type.category() == Type::Category::Array)
{
auto const& arrayType(dynamic_cast<ArrayType const&>(type));
2015-06-29 18:05:41 +00:00
m_isLValue = (
*m_memberName == "length" &&
arrayType.location() == DataLocation::Storage &&
arrayType.isDynamicallySized()
);
}
else
m_isLValue = false;
2014-10-13 16:22:15 +00:00
}
void IndexAccess::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
m_base->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
switch (m_base->type()->category())
2015-02-20 23:46:35 +00:00
{
case Type::Category::Array:
{
2015-08-31 16:44:29 +00:00
ArrayType const& type = dynamic_cast<ArrayType const&>(*m_base->type());
if (!m_index)
BOOST_THROW_EXCEPTION(createTypeError("Index expression cannot be omitted."));
2015-05-28 14:20:50 +00:00
if (type.isString())
BOOST_THROW_EXCEPTION(createTypeError("Index access for string is not possible."));
2015-02-20 23:46:35 +00:00
m_index->expectType(IntegerType(256));
m_type = type.baseType();
2015-09-10 08:17:17 +00:00
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."));
2015-06-17 10:01:39 +00:00
m_isLValue = type.location() != DataLocation::CallData;
2015-02-20 23:46:35 +00:00
break;
}
case Type::Category::Mapping:
{
2015-08-31 16:44:29 +00:00
MappingType const& type = dynamic_cast<MappingType const&>(*m_base->type());
if (!m_index)
BOOST_THROW_EXCEPTION(createTypeError("Index expression cannot be omitted."));
2015-08-31 16:44:29 +00:00
m_index->expectType(*type.keyType());
m_type = type.valueType();
2015-02-20 23:46:35 +00:00
m_isLValue = true;
break;
}
case Type::Category::TypeType:
{
2015-08-31 16:44:29 +00:00
TypeType const& type = dynamic_cast<TypeType const&>(*m_base->type());
if (!m_index)
2015-08-31 16:44:29 +00:00
m_type = make_shared<TypeType>(make_shared<ArrayType>(DataLocation::Memory, type.actualType()));
else
{
m_index->checkTypeRequirements(nullptr);
2015-08-31 16:44:29 +00:00
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(),
2015-06-17 10:01:39 +00:00
length->literalValue(nullptr)
));
}
break;
}
2015-02-20 23:46:35 +00:00
default:
BOOST_THROW_EXCEPTION(m_base->createTypeError(
"Indexed expression has to be a type, mapping or array (is " +
m_base->type()->toString() +
")"
));
2015-02-20 23:46:35 +00:00
}
2014-10-13 16:22:15 +00:00
}
void Identifier::checkTypeRequirements(TypePointers const* _argumentTypes)
2015-03-01 03:34:39 +00:00
{
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."));
2014-10-13 16:22:15 +00:00
}
2015-08-31 16:44:29 +00:00
Declaration const& Identifier::referencedDeclaration() const
{
solAssert(!!m_referencedDeclaration, "Identifier not resolved.");
return *m_referencedDeclaration;
}
void Identifier::overloadResolution(TypePointers const& _argumentTypes)
2015-03-01 03:34:39 +00:00
{
solAssert(!m_referencedDeclaration, "Referenced declaration should be null before overload resolution.");
solAssert(!m_overloadedDeclarations.empty(), "No candidates for overload resolution found.");
2015-03-01 03:34:39 +00:00
2015-05-04 15:18:01 +00:00
vector<Declaration const*> possibles;
if (m_overloadedDeclarations.size() == 1)
m_referencedDeclaration = *m_overloadedDeclarations.begin();
2015-03-01 03:34:39 +00:00
for (Declaration const* declaration: m_overloadedDeclarations)
2015-03-01 03:34:39 +00:00
{
2015-08-31 16:44:29 +00:00
TypePointer const& function = declaration->type();
auto const* functionType = dynamic_cast<FunctionType const*>(function.get());
if (functionType && functionType->canTakeArguments(_argumentTypes))
possibles.push_back(declaration);
2015-03-01 03:34:39 +00:00
}
if (possibles.size() == 1)
m_referencedDeclaration = possibles.front();
else if (possibles.empty())
BOOST_THROW_EXCEPTION(createTypeError("No matching declaration found after argument-dependent lookup."));
2015-03-01 03:34:39 +00:00
else
BOOST_THROW_EXCEPTION(createTypeError("No unique declaration found after argument-dependent lookup."));
2015-03-01 03:34:39 +00:00
}
void ElementaryTypeNameExpression::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
m_type = make_shared<TypeType>(Type::fromElementaryTypeName(m_typeToken));
2014-10-13 16:22:15 +00:00
}
void Literal::checkTypeRequirements(TypePointers const*)
2014-10-13 16:22:15 +00:00
{
m_type = Type::forLiteral(*this);
if (!m_type)
2014-12-19 10:31:17 +00:00
BOOST_THROW_EXCEPTION(createTypeError("Invalid literal value."));
2014-10-13 16:22:15 +00:00
}
2014-10-16 12:08:54 +00:00
}
}