2014-10-07 16:25:04 +00:00
|
|
|
/*
|
2014-10-16 12:08:54 +00:00
|
|
|
This file is part of cpp-ethereum.
|
2014-10-07 16:25:04 +00:00
|
|
|
|
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-07 16:25:04 +00:00
|
|
|
|
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-07 16:25:04 +00:00
|
|
|
|
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/>.
|
2014-10-07 16:25:04 +00:00
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @author Christian <c@ethdev.com>
|
|
|
|
* @date 2014
|
|
|
|
* Solidity abstract syntax tree.
|
|
|
|
*/
|
|
|
|
|
2014-10-13 16:22:15 +00:00
|
|
|
#include <algorithm>
|
2014-12-17 15:23:18 +00:00
|
|
|
#include <libsolidity/Utils.h>
|
2014-10-07 16:25:04 +00:00
|
|
|
#include <libsolidity/AST.h>
|
2014-10-10 14:37:54 +00:00
|
|
|
#include <libsolidity/ASTVisitor.h>
|
2014-10-15 12:45:51 +00:00
|
|
|
#include <libsolidity/Exceptions.h>
|
2014-12-08 12:33:13 +00:00
|
|
|
#include <libsolidity/AST_accept.h>
|
2014-10-10 14:37:54 +00:00
|
|
|
|
2015-01-07 15:39:21 +00:00
|
|
|
#include <libdevcrypto/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
|
|
|
|
{
|
2014-10-10 14:37:54 +00:00
|
|
|
|
2014-11-20 17:33:23 +00:00
|
|
|
TypeError ASTNode::createTypeError(string const& _description) const
|
2014-10-23 17:22:30 +00:00
|
|
|
{
|
|
|
|
return TypeError() << errinfo_sourceLocation(getLocation()) << errinfo_comment(_description);
|
|
|
|
}
|
|
|
|
|
2015-01-22 00:02:38 +00:00
|
|
|
TypePointer ContractDefinition::getType(ContractDefinition const* _currentContract) const
|
|
|
|
{
|
|
|
|
return make_shared<TypeType>(make_shared<ContractType>(*this), _currentContract);
|
|
|
|
}
|
|
|
|
|
2014-12-15 15:09:50 +00:00
|
|
|
void ContractDefinition::checkTypeRequirements()
|
|
|
|
{
|
2015-01-22 00:02:38 +00:00
|
|
|
for (ASTPointer<InheritanceSpecifier> const& baseSpecifier: getBaseContracts())
|
|
|
|
baseSpecifier->checkTypeRequirements();
|
2015-01-19 20:05:47 +00:00
|
|
|
|
2015-01-16 16:50:10 +00:00
|
|
|
checkIllegalOverrides();
|
|
|
|
|
2014-12-15 15:09:50 +00:00
|
|
|
FunctionDefinition const* constructor = getConstructor();
|
|
|
|
if (constructor && !constructor->getReturnParameters().empty())
|
|
|
|
BOOST_THROW_EXCEPTION(constructor->getReturnParameterList()->createTypeError(
|
2015-01-29 21:50:20 +00:00
|
|
|
"Non-empty \"returns\" directive for constructor."));
|
|
|
|
|
2015-02-09 01:06:30 +00:00
|
|
|
FunctionDefinition const* fallbackFunction = nullptr;
|
|
|
|
for (ASTPointer<FunctionDefinition> const& function: getDefinedFunctions())
|
|
|
|
if (function->getName().empty())
|
|
|
|
{
|
|
|
|
if (fallbackFunction)
|
|
|
|
BOOST_THROW_EXCEPTION(DeclarationError() << errinfo_comment("Only one fallback function is allowed."));
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fallbackFunction = function.get();
|
|
|
|
if (!fallbackFunction->getParameters().empty())
|
|
|
|
BOOST_THROW_EXCEPTION(fallbackFunction->getParameterList().createTypeError("Fallback function cannot take parameters."));
|
|
|
|
}
|
|
|
|
}
|
2015-01-22 00:02:38 +00:00
|
|
|
for (ASTPointer<ModifierDefinition> const& modifier: getFunctionModifiers())
|
|
|
|
modifier->checkTypeRequirements();
|
|
|
|
|
2014-12-15 15:09:50 +00:00
|
|
|
for (ASTPointer<FunctionDefinition> const& function: getDefinedFunctions())
|
|
|
|
function->checkTypeRequirements();
|
2015-01-14 09:16:58 +00:00
|
|
|
|
2015-02-17 15:21:38 +00:00
|
|
|
for (ASTPointer<VariableDeclaration> const& variable: m_stateVariables)
|
|
|
|
variable->checkTypeRequirements();
|
|
|
|
|
2015-01-14 09:16:58 +00:00
|
|
|
// check for hash collisions in function signatures
|
|
|
|
set<FixedHash<4>> hashes;
|
2015-01-29 15:39:30 +00:00
|
|
|
for (auto const& it: getInterfaceFunctionList())
|
2015-01-14 09:16:58 +00:00
|
|
|
{
|
2015-01-29 15:39:30 +00:00
|
|
|
FixedHash<4> const& hash = it.first;
|
2015-01-14 09:16:58 +00:00
|
|
|
if (hashes.count(hash))
|
2015-01-22 16:40:22 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError(
|
2015-01-23 15:37:06 +00:00
|
|
|
std::string("Function signature hash collision for ") +
|
2015-01-29 15:39:30 +00:00
|
|
|
it.second->getCanonicalSignature()));
|
2015-01-14 09:16:58 +00:00
|
|
|
hashes.insert(hash);
|
|
|
|
}
|
2014-12-15 15:09:50 +00:00
|
|
|
}
|
|
|
|
|
2015-01-29 15:48:39 +00:00
|
|
|
map<FixedHash<4>, FunctionTypePointer> ContractDefinition::getInterfaceFunctions() const
|
2014-11-11 16:41:48 +00:00
|
|
|
{
|
2015-01-23 15:37:06 +00:00
|
|
|
auto exportedFunctionList = getInterfaceFunctionList();
|
|
|
|
|
2015-01-29 15:48:39 +00:00
|
|
|
map<FixedHash<4>, FunctionTypePointer> exportedFunctions;
|
2015-01-23 15:37:06 +00:00
|
|
|
for (auto const& it: exportedFunctionList)
|
2015-01-29 15:39:30 +00:00
|
|
|
exportedFunctions.insert(it);
|
2015-01-22 16:40:22 +00:00
|
|
|
|
2015-01-14 09:16:58 +00:00
|
|
|
solAssert(exportedFunctionList.size() == exportedFunctions.size(),
|
|
|
|
"Hash collision at Function Definition Hash calculation");
|
2014-11-11 16:41:48 +00:00
|
|
|
|
|
|
|
return exportedFunctions;
|
|
|
|
}
|
|
|
|
|
2014-12-12 15:49:26 +00:00
|
|
|
FunctionDefinition const* ContractDefinition::getConstructor() const
|
|
|
|
{
|
|
|
|
for (ASTPointer<FunctionDefinition> const& f: m_definedFunctions)
|
2015-01-20 14:58:04 +00:00
|
|
|
if (f->isConstructor())
|
2014-12-12 15:49:26 +00:00
|
|
|
return f.get();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-01-29 21:50:20 +00:00
|
|
|
FunctionDefinition const* ContractDefinition::getFallbackFunction() const
|
|
|
|
{
|
|
|
|
for (ContractDefinition const* contract: getLinearizedBaseContracts())
|
|
|
|
for (ASTPointer<FunctionDefinition> const& f: contract->getDefinedFunctions())
|
|
|
|
if (f->getName().empty())
|
|
|
|
return f.get();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-01-16 16:50:10 +00:00
|
|
|
void ContractDefinition::checkIllegalOverrides() const
|
2015-01-14 09:16:58 +00:00
|
|
|
{
|
2015-01-22 00:02:38 +00:00
|
|
|
// TODO unify this at a later point. for this we need to put the constness and the access specifier
|
|
|
|
// into the types
|
2015-01-16 16:50:10 +00:00
|
|
|
map<string, FunctionDefinition const*> functions;
|
2015-01-22 00:02:38 +00:00
|
|
|
map<string, ModifierDefinition const*> modifiers;
|
2015-01-16 16:50:10 +00:00
|
|
|
|
|
|
|
// We search from derived to base, so the stored item causes the error.
|
|
|
|
for (ContractDefinition const* contract: getLinearizedBaseContracts())
|
2015-01-22 00:02:38 +00:00
|
|
|
{
|
2015-01-16 16:50:10 +00:00
|
|
|
for (ASTPointer<FunctionDefinition> const& function: contract->getDefinedFunctions())
|
2015-01-14 09:16:58 +00:00
|
|
|
{
|
2015-01-20 14:58:04 +00:00
|
|
|
if (function->isConstructor())
|
2015-01-22 00:02:38 +00:00
|
|
|
continue; // constructors can neither be overridden nor override anything
|
|
|
|
string const& name = function->getName();
|
|
|
|
if (modifiers.count(name))
|
|
|
|
BOOST_THROW_EXCEPTION(modifiers[name]->createTypeError("Override changes function to modifier."));
|
|
|
|
FunctionDefinition const*& override = functions[name];
|
2015-01-16 16:50:10 +00:00
|
|
|
if (!override)
|
|
|
|
override = function.get();
|
2015-02-02 16:24:09 +00:00
|
|
|
else if (override->getVisibility() != function->getVisibility() ||
|
2015-01-16 16:50:10 +00:00
|
|
|
override->isDeclaredConst() != function->isDeclaredConst() ||
|
|
|
|
FunctionType(*override) != FunctionType(*function))
|
|
|
|
BOOST_THROW_EXCEPTION(override->createTypeError("Override changes extended function signature."));
|
2015-01-14 09:16:58 +00:00
|
|
|
}
|
2015-01-22 00:02:38 +00:00
|
|
|
for (ASTPointer<ModifierDefinition> const& modifier: contract->getFunctionModifiers())
|
|
|
|
{
|
|
|
|
string const& name = modifier->getName();
|
|
|
|
if (functions.count(name))
|
|
|
|
BOOST_THROW_EXCEPTION(functions[name]->createTypeError("Override changes modifier to function."));
|
|
|
|
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."));
|
|
|
|
}
|
|
|
|
}
|
2015-01-16 16:50:10 +00:00
|
|
|
}
|
2015-01-14 09:16:58 +00:00
|
|
|
|
2015-01-31 13:41:11 +00:00
|
|
|
std::vector<ASTPointer<EventDefinition>> const& ContractDefinition::getInterfaceEvents() const
|
|
|
|
{
|
|
|
|
if (!m_interfaceEvents)
|
|
|
|
{
|
|
|
|
set<string> eventsSeen;
|
|
|
|
m_interfaceEvents.reset(new std::vector<ASTPointer<EventDefinition>>());
|
|
|
|
for (ContractDefinition const* contract: getLinearizedBaseContracts())
|
|
|
|
for (ASTPointer<EventDefinition> const& e: contract->getEvents())
|
|
|
|
if (eventsSeen.count(e->getName()) == 0)
|
|
|
|
{
|
|
|
|
eventsSeen.insert(e->getName());
|
|
|
|
m_interfaceEvents->push_back(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *m_interfaceEvents;
|
|
|
|
}
|
|
|
|
|
2015-01-29 16:28:14 +00:00
|
|
|
vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::getInterfaceFunctionList() const
|
2015-01-16 16:50:10 +00:00
|
|
|
{
|
|
|
|
if (!m_interfaceFunctionList)
|
|
|
|
{
|
|
|
|
set<string> functionsSeen;
|
2015-01-29 16:28:14 +00:00
|
|
|
m_interfaceFunctionList.reset(new vector<pair<FixedHash<4>, FunctionTypePointer>>());
|
2015-01-16 16:50:10 +00:00
|
|
|
for (ContractDefinition const* contract: getLinearizedBaseContracts())
|
2015-01-22 16:40:22 +00:00
|
|
|
{
|
2015-01-16 16:50:10 +00:00
|
|
|
for (ASTPointer<FunctionDefinition> const& f: contract->getDefinedFunctions())
|
2015-01-29 21:50:20 +00:00
|
|
|
if (f->isPublic() && !f->isConstructor() && !f->getName().empty() && functionsSeen.count(f->getName()) == 0)
|
2015-01-16 16:50:10 +00:00
|
|
|
{
|
|
|
|
functionsSeen.insert(f->getName());
|
|
|
|
FixedHash<4> hash(dev::sha3(f->getCanonicalSignature()));
|
2015-01-29 15:39:30 +00:00
|
|
|
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*f, false)));
|
2015-01-16 16:50:10 +00:00
|
|
|
}
|
2015-01-22 16:40:22 +00:00
|
|
|
|
|
|
|
for (ASTPointer<VariableDeclaration> const& v: contract->getStateVariables())
|
2015-01-28 17:06:45 +00:00
|
|
|
if (v->isPublic() && functionsSeen.count(v->getName()) == 0)
|
2015-01-22 16:40:22 +00:00
|
|
|
{
|
|
|
|
FunctionType ftype(*v);
|
|
|
|
functionsSeen.insert(v->getName());
|
2015-01-23 15:37:06 +00:00
|
|
|
FixedHash<4> hash(dev::sha3(ftype.getCanonicalSignature(v->getName())));
|
2015-01-29 15:39:30 +00:00
|
|
|
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*v)));
|
2015-01-22 16:40:22 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-16 16:50:10 +00:00
|
|
|
}
|
|
|
|
return *m_interfaceFunctionList;
|
2015-01-14 09:16:58 +00:00
|
|
|
}
|
|
|
|
|
2015-02-13 16:34:46 +00:00
|
|
|
TypePointer EnumValue::getType(ContractDefinition const*) const
|
2015-02-10 12:40:21 +00:00
|
|
|
{
|
2015-02-13 16:32:34 +00:00
|
|
|
EnumDefinition const* parentDef = dynamic_cast<EnumDefinition const*>(getScope());
|
2015-02-13 21:52:04 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2015-01-19 20:05:47 +00:00
|
|
|
void InheritanceSpecifier::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
m_baseName->checkTypeRequirements();
|
|
|
|
for (ASTPointer<Expression> const& argument: m_arguments)
|
|
|
|
argument->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())
|
|
|
|
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]))
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Invalid type for argument in constructer call."));
|
|
|
|
}
|
|
|
|
|
2015-01-22 00:02:38 +00:00
|
|
|
TypePointer StructDefinition::getType(ContractDefinition const*) const
|
|
|
|
{
|
|
|
|
return make_shared<TypeType>(make_shared<StructType>(*this));
|
|
|
|
}
|
|
|
|
|
2014-12-06 01:19:10 +00:00
|
|
|
void StructDefinition::checkMemberTypes() const
|
2014-12-03 06:46:55 +00:00
|
|
|
{
|
|
|
|
for (ASTPointer<VariableDeclaration> const& member: getMembers())
|
|
|
|
if (!member->getType()->canBeStored())
|
|
|
|
BOOST_THROW_EXCEPTION(member->createTypeError("Type cannot be used in struct."));
|
|
|
|
}
|
|
|
|
|
2014-12-06 01:19:10 +00:00
|
|
|
void StructDefinition::checkRecursion() const
|
2014-12-03 06:46:55 +00:00
|
|
|
{
|
|
|
|
set<StructDefinition const*> definitionsSeen;
|
|
|
|
vector<StructDefinition const*> queue = {this};
|
|
|
|
while (!queue.empty())
|
|
|
|
{
|
|
|
|
StructDefinition const* def = queue.back();
|
|
|
|
queue.pop_back();
|
|
|
|
if (definitionsSeen.count(def))
|
|
|
|
BOOST_THROW_EXCEPTION(ParserError() << errinfo_sourceLocation(def->getLocation())
|
|
|
|
<< errinfo_comment("Recursive struct definition."));
|
|
|
|
definitionsSeen.insert(def);
|
|
|
|
for (ASTPointer<VariableDeclaration> const& member: def->getMembers())
|
2015-02-09 13:00:12 +00:00
|
|
|
if (member->getType()->getCategory() == Type::Category::Struct)
|
2014-12-03 06:46:55 +00:00
|
|
|
{
|
2014-12-06 01:19:10 +00:00
|
|
|
UserDefinedTypeName const& typeName = dynamic_cast<UserDefinedTypeName const&>(*member->getTypeName());
|
2014-12-03 06:46:55 +00:00
|
|
|
queue.push_back(&dynamic_cast<StructDefinition const&>(*typeName.getReferencedDeclaration()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-09 17:08:56 +00:00
|
|
|
TypePointer EnumDefinition::getType(ContractDefinition const*) const
|
|
|
|
{
|
2015-02-11 15:37:46 +00:00
|
|
|
return make_shared<TypeType>(make_shared<EnumType>(*this));
|
2015-02-09 17:08:56 +00:00
|
|
|
}
|
|
|
|
|
2015-01-22 00:02:38 +00:00
|
|
|
TypePointer FunctionDefinition::getType(ContractDefinition const*) const
|
|
|
|
{
|
|
|
|
return make_shared<FunctionType>(*this);
|
|
|
|
}
|
|
|
|
|
2014-11-10 16:31:09 +00:00
|
|
|
void FunctionDefinition::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
for (ASTPointer<VariableDeclaration> const& var: getParameters() + getReturnParameters())
|
|
|
|
if (!var->getType()->canLiveOutsideStorage())
|
|
|
|
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
2015-01-22 00:02:38 +00:00
|
|
|
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
|
|
|
|
modifier->checkTypeRequirements();
|
2014-11-10 16:31:09 +00:00
|
|
|
|
|
|
|
m_body->checkTypeRequirements();
|
|
|
|
}
|
|
|
|
|
2015-01-07 09:45:59 +00:00
|
|
|
string FunctionDefinition::getCanonicalSignature() const
|
2015-01-06 16:42:38 +00:00
|
|
|
{
|
2015-01-22 16:40:22 +00:00
|
|
|
return FunctionType(*this).getCanonicalSignature(getName());
|
2015-01-06 16:42:38 +00:00
|
|
|
}
|
|
|
|
|
2015-02-14 00:22:44 +00:00
|
|
|
bool VariableDeclaration::isLValue() const
|
2015-01-22 00:02:38 +00:00
|
|
|
{
|
2015-02-19 16:43:53 +00:00
|
|
|
// External function parameters are Read-Only
|
|
|
|
return !isExternalFunctionParameter();
|
2015-02-14 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
2015-02-17 15:21:38 +00:00
|
|
|
void VariableDeclaration::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
if (m_value)
|
|
|
|
m_value->checkTypeRequirements();
|
|
|
|
}
|
|
|
|
|
2015-02-19 16:43:53 +00:00
|
|
|
bool VariableDeclaration::isExternalFunctionParameter() const
|
2015-02-14 00:22:44 +00:00
|
|
|
{
|
|
|
|
auto const* function = dynamic_cast<FunctionDefinition const*>(getScope());
|
2015-02-19 16:43:53 +00:00
|
|
|
if (!function || function->getVisibility() != Declaration::Visibility::External)
|
2015-02-14 00:22:44 +00:00
|
|
|
return false;
|
|
|
|
for (auto const& variable: function->getParameters())
|
|
|
|
if (variable.get() == this)
|
|
|
|
return true;
|
|
|
|
return false;
|
2015-01-22 00:02:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TypePointer ModifierDefinition::getType(ContractDefinition const*) const
|
|
|
|
{
|
|
|
|
return make_shared<ModifierType>(*this);
|
|
|
|
}
|
|
|
|
|
2015-01-21 10:16:18 +00:00
|
|
|
void ModifierDefinition::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
m_body->checkTypeRequirements();
|
|
|
|
}
|
|
|
|
|
2015-01-22 00:02:38 +00:00
|
|
|
void ModifierInvocation::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
m_modifierName->checkTypeRequirements();
|
|
|
|
for (ASTPointer<Expression> const& argument: m_arguments)
|
|
|
|
argument->checkTypeRequirements();
|
|
|
|
|
|
|
|
ModifierDefinition const* modifier = dynamic_cast<ModifierDefinition const*>(m_modifierName->getReferencedDeclaration());
|
|
|
|
solAssert(modifier, "Function modifier not found.");
|
|
|
|
vector<ASTPointer<VariableDeclaration>> const& parameters = modifier->getParameters();
|
|
|
|
if (parameters.size() != m_arguments.size())
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Wrong argument count for modifier invocation."));
|
|
|
|
for (size_t i = 0; i < m_arguments.size(); ++i)
|
|
|
|
if (!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*parameters[i]->getType()))
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Invalid type for argument in modifier invocation."));
|
|
|
|
}
|
|
|
|
|
2015-01-29 13:35:28 +00:00
|
|
|
void EventDefinition::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
int numIndexed = 0;
|
|
|
|
for (ASTPointer<VariableDeclaration> const& var: getParameters())
|
|
|
|
{
|
|
|
|
if (var->isIndexed())
|
|
|
|
numIndexed++;
|
|
|
|
if (!var->getType()->canLiveOutsideStorage())
|
|
|
|
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
|
|
|
}
|
|
|
|
if (numIndexed > 3)
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("More than 3 indexed arguments for event."));
|
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +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();
|
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void IfStatement::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-10-30 00:20:32 +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
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void WhileStatement::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-10-30 00:20:32 +00:00
|
|
|
m_condition->expectType(BoolType());
|
2014-10-13 16:22:15 +00:00
|
|
|
m_body->checkTypeRequirements();
|
|
|
|
}
|
|
|
|
|
2014-12-15 16:45:18 +00:00
|
|
|
void ForStatement::checkTypeRequirements()
|
|
|
|
{
|
2014-12-16 12:20:50 +00:00
|
|
|
if (m_initExpression)
|
|
|
|
m_initExpression->checkTypeRequirements();
|
|
|
|
if (m_condExpression)
|
|
|
|
m_condExpression->expectType(BoolType());
|
|
|
|
if (m_loopExpression)
|
|
|
|
m_loopExpression->checkTypeRequirements();
|
2014-12-15 16:45:18 +00:00
|
|
|
m_body->checkTypeRequirements();
|
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void Return::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-10-30 00:16:11 +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."));
|
2014-10-13 16:22:15 +00:00
|
|
|
if (m_returnParameters->getParameters().size() != 1)
|
2014-10-23 17:22:30 +00:00
|
|
|
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
|
2014-10-30 00:20:32 +00:00
|
|
|
m_expression->expectType(*m_returnParameters->getParameters().front()->getType());
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2015-02-17 15:21:38 +00:00
|
|
|
void VariableDeclarationStatement::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
|
|
|
// Variables can be declared without type (with "var"), in which case the first assignment
|
2014-11-10 16:31:09 +00:00
|
|
|
// sets the type.
|
2014-10-13 16:22:15 +00:00
|
|
|
// Note that assignments before the first declaration are legal because of the special scoping
|
|
|
|
// rules inherited from JavaScript.
|
2015-02-17 15:21:38 +00:00
|
|
|
if (m_variable->getValue())
|
2014-10-16 12:08:54 +00:00
|
|
|
{
|
|
|
|
if (m_variable->getType())
|
2015-02-17 15:21:38 +00:00
|
|
|
m_variable->getValue()->expectType(*m_variable->getType());
|
2014-10-16 12:08:54 +00:00
|
|
|
else
|
2014-10-20 12:00:37 +00:00
|
|
|
{
|
2014-10-13 16:22:15 +00:00
|
|
|
// no type declared and no previous assignment, infer the type
|
2015-02-17 15:21:38 +00:00
|
|
|
m_variable->getValue()->checkTypeRequirements();
|
|
|
|
TypePointer type = m_variable->getValue()->getType();
|
2015-02-09 13:00:12 +00:00
|
|
|
if (type->getCategory() == Type::Category::IntegerConstant)
|
2014-12-19 10:31:17 +00:00
|
|
|
{
|
|
|
|
auto intType = dynamic_pointer_cast<IntegerConstantType const>(type)->getIntegerType();
|
|
|
|
if (!intType)
|
2015-02-17 15:21:38 +00:00
|
|
|
BOOST_THROW_EXCEPTION(m_variable->getValue()->createTypeError("Invalid integer constant " + type->toString()));
|
2014-12-19 10:31:17 +00:00
|
|
|
type = intType;
|
|
|
|
}
|
2015-02-09 13:00:12 +00:00
|
|
|
else if (type->getCategory() == Type::Category::Void)
|
2015-02-05 23:44:38 +00:00
|
|
|
BOOST_THROW_EXCEPTION(m_variable->createTypeError("var cannot be void type"));
|
2014-12-19 10:31:17 +00:00
|
|
|
m_variable->setType(type);
|
2014-10-20 12:00:37 +00:00
|
|
|
}
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
}
|
2014-10-20 12:00:37 +00:00
|
|
|
void Assignment::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-10-20 12:00:37 +00:00
|
|
|
m_leftHandSide->checkTypeRequirements();
|
2014-11-10 16:31:09 +00:00
|
|
|
m_leftHandSide->requireLValue();
|
2015-02-12 17:38:07 +00:00
|
|
|
if (m_leftHandSide->getType()->getCategory() == Type::Category::Mapping)
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Mappings cannot be assigned to."));
|
2014-10-13 16:22:15 +00:00
|
|
|
m_type = m_leftHandSide->getType();
|
2015-02-09 13:00:12 +00:00
|
|
|
if (m_assigmentOperator == Token::Assign)
|
2014-12-18 17:53:43 +00:00
|
|
|
m_rightHandSide->expectType(*m_type);
|
|
|
|
else
|
|
|
|
{
|
2014-10-20 10:41:56 +00:00
|
|
|
// compound assignment
|
2014-12-18 17:53:43 +00:00
|
|
|
m_rightHandSide->checkTypeRequirements();
|
2015-01-06 17:55:31 +00:00
|
|
|
TypePointer resultType = m_type->binaryOperatorResult(Token::AssignmentToBinaryOp(m_assigmentOperator),
|
|
|
|
m_rightHandSide->getType());
|
2014-12-18 17:53:43 +00:00
|
|
|
if (!resultType || *resultType != *m_type)
|
2014-12-19 10:31:17 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Operator " + string(Token::toString(m_assigmentOperator)) +
|
|
|
|
" not compatible with types " +
|
|
|
|
m_type->toString() + " and " +
|
|
|
|
m_rightHandSide->getType()->toString()));
|
2014-12-18 17:53:43 +00:00
|
|
|
}
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-30 00:20:32 +00:00
|
|
|
void ExpressionStatement::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
m_expression->checkTypeRequirements();
|
2015-02-09 13:00:12 +00:00
|
|
|
if (m_expression->getType()->getCategory() == Type::Category::IntegerConstant)
|
2014-12-19 10:31:17 +00:00
|
|
|
if (!dynamic_pointer_cast<IntegerConstantType const>(m_expression->getType())->getIntegerType())
|
|
|
|
BOOST_THROW_EXCEPTION(m_expression->createTypeError("Invalid integer constant."));
|
2014-10-30 00:20:32 +00:00
|
|
|
}
|
|
|
|
|
2014-11-05 14:04:33 +00:00
|
|
|
void Expression::expectType(Type const& _expectedType)
|
2014-10-30 00:20:32 +00:00
|
|
|
{
|
|
|
|
checkTypeRequirements();
|
2014-11-06 21:04:10 +00:00
|
|
|
Type const& type = *getType();
|
2014-11-04 14:29:08 +00:00
|
|
|
if (!type.isImplicitlyConvertibleTo(_expectedType))
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Type " + type.toString() +
|
|
|
|
" not implicitly convertible to expected type "
|
|
|
|
+ _expectedType.toString() + "."));
|
2014-10-30 00:20:32 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 16:31:09 +00:00
|
|
|
void Expression::requireLValue()
|
|
|
|
{
|
2014-12-01 16:32:10 +00:00
|
|
|
if (!isLValue())
|
2014-11-10 16:31:09 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Expression has to be an lvalue."));
|
|
|
|
m_lvalueRequested = true;
|
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void UnaryOperation::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2015-02-10 08:52:19 +00:00
|
|
|
// Inc, Dec, Add, Sub, Not, BitNot, Delete
|
2014-10-20 12:00:37 +00:00
|
|
|
m_subExpression->checkTypeRequirements();
|
2015-02-09 13:00:12 +00:00
|
|
|
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-01-06 18:08:24 +00:00
|
|
|
m_type = m_subExpression->getType()->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
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void BinaryOperation::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
|
|
|
m_left->checkTypeRequirements();
|
2014-11-05 22:35:00 +00:00
|
|
|
m_right->checkTypeRequirements();
|
2015-01-06 17:55:31 +00:00
|
|
|
m_commonType = m_left->getType()->binaryOperatorResult(m_operator, m_right->getType());
|
2014-12-18 17:53:43 +00:00
|
|
|
if (!m_commonType)
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Operator " + string(Token::toString(m_operator)) +
|
|
|
|
" not compatible with types " +
|
|
|
|
m_left->getType()->toString() + " and " +
|
2014-11-04 14:29:08 +00:00
|
|
|
m_right->getType()->toString()));
|
2014-12-18 17:53:43 +00:00
|
|
|
m_type = Token::isCompareOp(m_operator) ? make_shared<BoolType>() : m_commonType;
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void FunctionCall::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
|
|
|
m_expression->checkTypeRequirements();
|
2014-10-20 12:00:37 +00:00
|
|
|
for (ASTPointer<Expression> const& argument: m_arguments)
|
2014-10-13 16:22:15 +00:00
|
|
|
argument->checkTypeRequirements();
|
2014-10-20 10:41:56 +00:00
|
|
|
|
|
|
|
Type const* expressionType = m_expression->getType().get();
|
|
|
|
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
|
|
|
//@todo for structs, we have to check the number of arguments to be equal to the
|
|
|
|
// number of non-mapping members
|
|
|
|
if (m_arguments.size() != 1)
|
2015-01-28 12:39:04 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("More than one argument for explicit type conversion."));
|
2015-01-29 17:26:00 +00:00
|
|
|
if (!m_names.empty())
|
2015-02-08 23:49:35 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Type conversion cannot allow named arguments."));
|
2014-11-05 13:20:56 +00:00
|
|
|
if (!m_arguments.front()->getType()->isExplicitlyConvertibleTo(*type.getActualType()))
|
2014-10-23 17:22:30 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Explicit type conversion not allowed."));
|
2014-11-05 13:20:56 +00:00
|
|
|
m_type = type.getActualType();
|
2014-10-16 12:08:54 +00:00
|
|
|
}
|
2014-12-12 15:49:26 +00:00
|
|
|
else if (FunctionType const* functionType = dynamic_cast<FunctionType const*>(expressionType))
|
2014-10-16 12:08:54 +00:00
|
|
|
{
|
2014-10-13 16:22:15 +00:00
|
|
|
//@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
|
2014-12-12 15:49:26 +00:00
|
|
|
TypePointers const& parameterTypes = functionType->getParameterTypes();
|
2015-02-10 09:45:57 +00:00
|
|
|
if (!functionType->takesArbitraryParameters() && parameterTypes.size() != m_arguments.size())
|
2014-10-23 17:22:30 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Wrong argument count for function call."));
|
2015-01-29 17:26:00 +00:00
|
|
|
|
|
|
|
if (m_names.empty())
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < m_arguments.size(); ++i)
|
2015-02-10 09:45:57 +00:00
|
|
|
if (!functionType->takesArbitraryParameters() &&
|
2015-02-08 23:49:35 +00:00
|
|
|
!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*parameterTypes[i]))
|
|
|
|
BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError("Invalid type for argument in function call."));
|
2015-01-29 17:26:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-02-10 09:45:57 +00:00
|
|
|
if (functionType->takesArbitraryParameters())
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Named arguments cannnot be used for functions "
|
|
|
|
"that take arbitrary parameters."));
|
2015-01-29 17:26:00 +00:00
|
|
|
auto const& parameterNames = functionType->getParameterNames();
|
|
|
|
if (parameterNames.size() != m_names.size())
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Some argument names are missing."));
|
|
|
|
|
|
|
|
// check duplicate names
|
2015-02-08 23:49:35 +00:00
|
|
|
for (size_t i = 0; i < m_names.size(); i++)
|
|
|
|
for (size_t j = i + 1; j < m_names.size(); j++)
|
2015-02-04 17:11:53 +00:00
|
|
|
if (*m_names[i] == *m_names[j])
|
2015-02-08 23:49:35 +00:00
|
|
|
BOOST_THROW_EXCEPTION(m_arguments[i]->createTypeError("Duplicate named argument."));
|
2015-01-29 17:26:00 +00:00
|
|
|
|
|
|
|
for (size_t i = 0; i < m_names.size(); i++) {
|
|
|
|
bool found = false;
|
|
|
|
for (size_t j = 0; j < parameterNames.size(); j++) {
|
2015-02-03 20:25:08 +00:00
|
|
|
if (parameterNames[j] == *m_names[i]) {
|
2015-01-29 17:26:00 +00:00
|
|
|
// check type convertible
|
|
|
|
if (!m_arguments[i]->getType()->isImplicitlyConvertibleTo(*parameterTypes[j]))
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Invalid type for argument in function call."));
|
|
|
|
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found)
|
2015-02-08 23:49:35 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Named argument does not match function declaration."));
|
2015-01-29 17:26:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-13 16:22:15 +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 structs
|
2014-12-12 15:49:26 +00:00
|
|
|
if (functionType->getReturnParameterTypes().empty())
|
2014-10-24 17:06:30 +00:00
|
|
|
m_type = make_shared<VoidType>();
|
2014-10-13 16:22:15 +00:00
|
|
|
else
|
2014-12-12 15:49:26 +00:00
|
|
|
m_type = functionType->getReturnParameterTypes().front();
|
2014-10-16 12:08:54 +00:00
|
|
|
}
|
2014-12-12 15:49:26 +00:00
|
|
|
else
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Type is not callable."));
|
2014-10-20 10:41:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FunctionCall::isTypeConversion() const
|
|
|
|
{
|
2015-02-10 08:52:19 +00:00
|
|
|
return m_expression->getType()->getCategory() == Type::Category::TypeType;
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-12-12 15:49:26 +00:00
|
|
|
void NewExpression::checkTypeRequirements()
|
|
|
|
{
|
|
|
|
m_contractName->checkTypeRequirements();
|
|
|
|
m_contract = dynamic_cast<ContractDefinition const*>(m_contractName->getReferencedDeclaration());
|
|
|
|
if (!m_contract)
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Identifier is not a contract."));
|
2015-01-13 17:12:19 +00:00
|
|
|
shared_ptr<ContractType const> contractType = make_shared<ContractType>(*m_contract);
|
|
|
|
TypePointers const& parameterTypes = contractType->getConstructorType()->getParameterTypes();
|
|
|
|
m_type = make_shared<FunctionType>(parameterTypes, TypePointers{contractType},
|
2015-02-09 13:08:48 +00:00
|
|
|
FunctionType::Location::Creation);
|
2014-12-12 15:49:26 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void MemberAccess::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-11-13 00:12:57 +00:00
|
|
|
m_expression->checkTypeRequirements();
|
2014-11-20 09:19:43 +00:00
|
|
|
Type const& type = *m_expression->getType();
|
|
|
|
m_type = type.getMemberType(*m_memberName);
|
|
|
|
if (!m_type)
|
2015-01-19 18:18:34 +00:00
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Member \"" + *m_memberName + "\" not found or not "
|
|
|
|
"visible in " + type.toString()));
|
2015-02-14 00:22:44 +00:00
|
|
|
m_isLValue = (type.getCategory() == Type::Category::Struct);
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void IndexAccess::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-11-10 16:31:09 +00:00
|
|
|
m_base->checkTypeRequirements();
|
2015-02-20 23:46:35 +00:00
|
|
|
switch (m_base->getType()->getCategory())
|
|
|
|
{
|
|
|
|
case Type::Category::Array:
|
|
|
|
{
|
|
|
|
ArrayType const& type = dynamic_cast<ArrayType const&>(*m_base->getType());
|
|
|
|
m_index->expectType(IntegerType(256));
|
|
|
|
m_type = type.getBaseType();
|
|
|
|
m_isLValue = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Type::Category::Mapping:
|
|
|
|
{
|
|
|
|
MappingType const& type = dynamic_cast<MappingType const&>(*m_base->getType());
|
|
|
|
m_index->expectType(*type.getKeyType());
|
|
|
|
m_type = type.getValueType();
|
|
|
|
m_isLValue = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
BOOST_THROW_EXCEPTION(m_base->createTypeError(
|
|
|
|
"Indexed expression has to be a mapping or array (is " + m_base->getType()->toString() + ")"));
|
|
|
|
}
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void Identifier::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2014-12-17 15:23:18 +00:00
|
|
|
solAssert(m_referencedDeclaration, "Identifier not resolved.");
|
2014-11-05 13:20:56 +00:00
|
|
|
|
2015-02-14 00:22:44 +00:00
|
|
|
m_isLValue = m_referencedDeclaration->isLValue();
|
2015-01-22 00:02:38 +00:00
|
|
|
m_type = m_referencedDeclaration->getType(m_currentContract);
|
|
|
|
if (!m_type)
|
|
|
|
BOOST_THROW_EXCEPTION(createTypeError("Declaration referenced before type could be determined."));
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void ElementaryTypeNameExpression::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
2015-01-07 20:35:35 +00:00
|
|
|
m_type = make_shared<TypeType>(Type::fromElementaryTypeName(m_typeToken));
|
2014-10-13 16:22:15 +00:00
|
|
|
}
|
|
|
|
|
2014-10-20 12:00:37 +00:00
|
|
|
void Literal::checkTypeRequirements()
|
2014-10-13 16:22:15 +00:00
|
|
|
{
|
|
|
|
m_type = Type::forLiteral(*this);
|
2014-11-04 12:24:35 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|