mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #1338 from LianaHus/sol_ConvertContractTypesToAddress
Solidity: Convert contract types to addresses
This commit is contained in:
commit
f728a336b1
22
AST.cpp
22
AST.cpp
@ -88,7 +88,7 @@ void ContractDefinition::checkTypeRequirements()
|
|||||||
if (hashes.count(hash))
|
if (hashes.count(hash))
|
||||||
BOOST_THROW_EXCEPTION(createTypeError(
|
BOOST_THROW_EXCEPTION(createTypeError(
|
||||||
std::string("Function signature hash collision for ") +
|
std::string("Function signature hash collision for ") +
|
||||||
it.second->getCanonicalSignature()));
|
it.second->externalSignature()));
|
||||||
hashes.insert(hash);
|
hashes.insert(hash);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -192,7 +192,7 @@ vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::getIn
|
|||||||
if (functionsSeen.count(f->getName()) == 0 && f->isPartOfExternalInterface())
|
if (functionsSeen.count(f->getName()) == 0 && f->isPartOfExternalInterface())
|
||||||
{
|
{
|
||||||
functionsSeen.insert(f->getName());
|
functionsSeen.insert(f->getName());
|
||||||
FixedHash<4> hash(dev::sha3(f->getCanonicalSignature()));
|
FixedHash<4> hash(dev::sha3(f->externalSignature()));
|
||||||
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*f, false)));
|
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*f, false)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -200,8 +200,9 @@ vector<pair<FixedHash<4>, FunctionTypePointer>> const& ContractDefinition::getIn
|
|||||||
if (functionsSeen.count(v->getName()) == 0 && v->isPartOfExternalInterface())
|
if (functionsSeen.count(v->getName()) == 0 && v->isPartOfExternalInterface())
|
||||||
{
|
{
|
||||||
FunctionType ftype(*v);
|
FunctionType ftype(*v);
|
||||||
|
solAssert(v->getType().get(), "");
|
||||||
functionsSeen.insert(v->getName());
|
functionsSeen.insert(v->getName());
|
||||||
FixedHash<4> hash(dev::sha3(ftype.getCanonicalSignature(v->getName())));
|
FixedHash<4> hash(dev::sha3(ftype.externalSignature(v->getName())));
|
||||||
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*v)));
|
m_interfaceFunctionList->push_back(make_pair(hash, make_shared<FunctionType>(*v)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -305,8 +306,12 @@ TypePointer FunctionDefinition::getType(ContractDefinition const*) const
|
|||||||
void FunctionDefinition::checkTypeRequirements()
|
void FunctionDefinition::checkTypeRequirements()
|
||||||
{
|
{
|
||||||
for (ASTPointer<VariableDeclaration> const& var: getParameters() + getReturnParameters())
|
for (ASTPointer<VariableDeclaration> const& var: getParameters() + getReturnParameters())
|
||||||
|
{
|
||||||
if (!var->getType()->canLiveOutsideStorage())
|
if (!var->getType()->canLiveOutsideStorage())
|
||||||
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
||||||
|
if (!(var->getType()->externalType()) && getVisibility() >= Visibility::Public)
|
||||||
|
BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed for function with external visibility"));
|
||||||
|
}
|
||||||
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
|
for (ASTPointer<ModifierInvocation> const& modifier: m_functionModifiers)
|
||||||
modifier->checkTypeRequirements(isConstructor() ?
|
modifier->checkTypeRequirements(isConstructor() ?
|
||||||
dynamic_cast<ContractDefinition const&>(*getScope()).getBaseContracts() :
|
dynamic_cast<ContractDefinition const&>(*getScope()).getBaseContracts() :
|
||||||
@ -315,9 +320,9 @@ void FunctionDefinition::checkTypeRequirements()
|
|||||||
m_body->checkTypeRequirements();
|
m_body->checkTypeRequirements();
|
||||||
}
|
}
|
||||||
|
|
||||||
string FunctionDefinition::getCanonicalSignature() const
|
string FunctionDefinition::externalSignature() const
|
||||||
{
|
{
|
||||||
return FunctionType(*this).getCanonicalSignature(getName());
|
return FunctionType(*this).externalSignature(getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool VariableDeclaration::isLValue() const
|
bool VariableDeclaration::isLValue() const
|
||||||
@ -342,8 +347,11 @@ void VariableDeclaration::checkTypeRequirements()
|
|||||||
if (!m_value)
|
if (!m_value)
|
||||||
return;
|
return;
|
||||||
if (m_type)
|
if (m_type)
|
||||||
|
{
|
||||||
m_value->expectType(*m_type);
|
m_value->expectType(*m_type);
|
||||||
else
|
if (m_isStateVariable && !m_type->externalType() && getVisibility() >= Visibility::Public)
|
||||||
|
BOOST_THROW_EXCEPTION(createTypeError("Internal type is not allowed for state variables."));
|
||||||
|
} else
|
||||||
{
|
{
|
||||||
// no type declared and no previous assignment, infer the type
|
// no type declared and no previous assignment, infer the type
|
||||||
m_value->checkTypeRequirements();
|
m_value->checkTypeRequirements();
|
||||||
@ -422,6 +430,8 @@ void EventDefinition::checkTypeRequirements()
|
|||||||
numIndexed++;
|
numIndexed++;
|
||||||
if (!var->getType()->canLiveOutsideStorage())
|
if (!var->getType()->canLiveOutsideStorage())
|
||||||
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
BOOST_THROW_EXCEPTION(var->createTypeError("Type is required to live outside storage."));
|
||||||
|
if (!var->getType()->externalType())
|
||||||
|
BOOST_THROW_EXCEPTION(var->createTypeError("Internal type is not allowed as event parameter type."));
|
||||||
}
|
}
|
||||||
if (numIndexed > 3)
|
if (numIndexed > 3)
|
||||||
BOOST_THROW_EXCEPTION(createTypeError("More than 3 indexed arguments for event."));
|
BOOST_THROW_EXCEPTION(createTypeError("More than 3 indexed arguments for event."));
|
||||||
|
4
AST.h
4
AST.h
@ -421,10 +421,10 @@ public:
|
|||||||
/// Checks that all parameters have allowed types and calls checkTypeRequirements on the body.
|
/// Checks that all parameters have allowed types and calls checkTypeRequirements on the body.
|
||||||
void checkTypeRequirements();
|
void checkTypeRequirements();
|
||||||
|
|
||||||
/// @returns the canonical signature of the function
|
/// @returns the external signature of the function
|
||||||
/// That consists of the name of the function followed by the types of the
|
/// That consists of the name of the function followed by the types of the
|
||||||
/// arguments separated by commas all enclosed in parentheses without any spaces.
|
/// arguments separated by commas all enclosed in parentheses without any spaces.
|
||||||
std::string getCanonicalSignature() const;
|
std::string externalSignature() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool m_isConstructor;
|
bool m_isConstructor;
|
||||||
|
@ -544,7 +544,7 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
|
|||||||
}
|
}
|
||||||
if (!event.isAnonymous())
|
if (!event.isAnonymous())
|
||||||
{
|
{
|
||||||
m_context << u256(h256::Arith(dev::sha3(function.getCanonicalSignature(event.getName()))));
|
m_context << u256(h256::Arith(dev::sha3(function.externalSignature(event.getName()))));
|
||||||
++numIndexed;
|
++numIndexed;
|
||||||
}
|
}
|
||||||
solAssert(numIndexed <= 4, "Too many indexed arguments.");
|
solAssert(numIndexed <= 4, "Too many indexed arguments.");
|
||||||
|
@ -129,7 +129,7 @@ std::unique_ptr<std::string> InterfaceHandler::getUserDocumentation(ContractDefi
|
|||||||
if (!m_notice.empty())
|
if (!m_notice.empty())
|
||||||
{// since @notice is the only user tag if missing function should not appear
|
{// since @notice is the only user tag if missing function should not appear
|
||||||
user["notice"] = Json::Value(m_notice);
|
user["notice"] = Json::Value(m_notice);
|
||||||
methods[it.second->getCanonicalSignature()] = user;
|
methods[it.second->externalSignature()] = user;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -185,7 +185,7 @@ std::unique_ptr<std::string> InterfaceHandler::getDevDocumentation(ContractDefin
|
|||||||
method["return"] = m_return;
|
method["return"] = m_return;
|
||||||
|
|
||||||
if (!method.empty()) // add the function, only if we have any documentation to add
|
if (!method.empty()) // add the function, only if we have any documentation to add
|
||||||
methods[it.second->getCanonicalSignature()] = method;
|
methods[it.second->externalSignature()] = method;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
doc["methods"] = methods;
|
doc["methods"] = methods;
|
||||||
|
40
Types.cpp
40
Types.cpp
@ -742,6 +742,23 @@ string ArrayType::toString() const
|
|||||||
return ret + "]";
|
return ret + "]";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TypePointer ArrayType::externalType() const
|
||||||
|
{
|
||||||
|
if (m_location != Location::CallData)
|
||||||
|
return TypePointer();
|
||||||
|
if (m_isByteArray)
|
||||||
|
return shared_from_this();
|
||||||
|
if (!m_baseType->externalType())
|
||||||
|
return TypePointer();
|
||||||
|
if (m_baseType->getCategory() == Category::Array && m_baseType->isDynamicallySized())
|
||||||
|
return TypePointer();
|
||||||
|
|
||||||
|
if (isDynamicallySized())
|
||||||
|
return std::make_shared<ArrayType>(Location::CallData, m_baseType->externalType());
|
||||||
|
else
|
||||||
|
return std::make_shared<ArrayType>(Location::CallData, m_baseType->externalType(), m_length);
|
||||||
|
}
|
||||||
|
|
||||||
shared_ptr<ArrayType> ArrayType::copyForLocation(ArrayType::Location _location) const
|
shared_ptr<ArrayType> ArrayType::copyForLocation(ArrayType::Location _location) const
|
||||||
{
|
{
|
||||||
auto copy = make_shared<ArrayType>(_location);
|
auto copy = make_shared<ArrayType>(_location);
|
||||||
@ -1081,6 +1098,19 @@ unsigned FunctionType::getSizeOnStack() const
|
|||||||
return size;
|
return size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TypePointer FunctionType::externalType() const
|
||||||
|
{
|
||||||
|
TypePointers paramTypes;
|
||||||
|
TypePointers retParamTypes;
|
||||||
|
|
||||||
|
for (auto it = m_parameterTypes.cbegin(); it != m_parameterTypes.cend(); ++it)
|
||||||
|
paramTypes.push_back((*it)->externalType());
|
||||||
|
for (auto it = m_returnParameterTypes.cbegin(); it != m_returnParameterTypes.cend(); ++it)
|
||||||
|
retParamTypes.push_back((*it)->externalType());
|
||||||
|
|
||||||
|
return make_shared<FunctionType>(paramTypes, retParamTypes, m_location, m_arbitraryParameters);
|
||||||
|
}
|
||||||
|
|
||||||
MemberList const& FunctionType::getMembers() const
|
MemberList const& FunctionType::getMembers() const
|
||||||
{
|
{
|
||||||
switch (m_location)
|
switch (m_location)
|
||||||
@ -1110,7 +1140,7 @@ MemberList const& FunctionType::getMembers() const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
string FunctionType::getCanonicalSignature(std::string const& _name) const
|
string FunctionType::externalSignature(std::string const& _name) const
|
||||||
{
|
{
|
||||||
std::string funcName = _name;
|
std::string funcName = _name;
|
||||||
if (_name == "")
|
if (_name == "")
|
||||||
@ -1120,8 +1150,12 @@ string FunctionType::getCanonicalSignature(std::string const& _name) const
|
|||||||
}
|
}
|
||||||
string ret = funcName + "(";
|
string ret = funcName + "(";
|
||||||
|
|
||||||
for (auto it = m_parameterTypes.cbegin(); it != m_parameterTypes.cend(); ++it)
|
TypePointers externalParameterTypes = dynamic_cast<FunctionType const&>(*externalType()).getParameterTypes();
|
||||||
ret += (*it)->toString() + (it + 1 == m_parameterTypes.cend() ? "" : ",");
|
for (auto it = externalParameterTypes.cbegin(); it != externalParameterTypes.cend(); ++it)
|
||||||
|
{
|
||||||
|
solAssert(!!(*it), "Parameter should have external type");
|
||||||
|
ret += (*it)->toString() + (it + 1 == externalParameterTypes.cend() ? "" : ",");
|
||||||
|
}
|
||||||
|
|
||||||
return ret + ")";
|
return ret + ")";
|
||||||
}
|
}
|
||||||
|
22
Types.h
22
Types.h
@ -187,6 +187,10 @@ public:
|
|||||||
"for type without literals."));
|
"for type without literals."));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// @returns a type suitable for outside of Solidity, i.e. for contract types it returns address.
|
||||||
|
/// If there is no such type, returns an empty shared pointer.
|
||||||
|
virtual TypePointer externalType() const { return TypePointer(); }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/// Convenience object used when returning an empty member list.
|
/// Convenience object used when returning an empty member list.
|
||||||
static const MemberList EmptyMemberList;
|
static const MemberList EmptyMemberList;
|
||||||
@ -217,10 +221,12 @@ public:
|
|||||||
virtual unsigned getStorageBytes() const override { return m_bits / 8; }
|
virtual unsigned getStorageBytes() const override { return m_bits / 8; }
|
||||||
virtual bool isValueType() const override { return true; }
|
virtual bool isValueType() const override { return true; }
|
||||||
|
|
||||||
virtual MemberList const& getMembers() const { return isAddress() ? AddressMemberList : EmptyMemberList; }
|
virtual MemberList const& getMembers() const override { return isAddress() ? AddressMemberList : EmptyMemberList; }
|
||||||
|
|
||||||
virtual std::string toString() const override;
|
virtual std::string toString() const override;
|
||||||
|
|
||||||
|
virtual TypePointer externalType() const override { return shared_from_this(); }
|
||||||
|
|
||||||
int getNumBits() const { return m_bits; }
|
int getNumBits() const { return m_bits; }
|
||||||
bool isAddress() const { return m_modifier == Modifier::Address; }
|
bool isAddress() const { return m_modifier == Modifier::Address; }
|
||||||
bool isSigned() const { return m_modifier == Modifier::Signed; }
|
bool isSigned() const { return m_modifier == Modifier::Signed; }
|
||||||
@ -292,6 +298,7 @@ public:
|
|||||||
|
|
||||||
virtual std::string toString() const override { return "bytes" + dev::toString(m_bytes); }
|
virtual std::string toString() const override { return "bytes" + dev::toString(m_bytes); }
|
||||||
virtual u256 literalValue(Literal const* _literal) const override;
|
virtual u256 literalValue(Literal const* _literal) const override;
|
||||||
|
virtual TypePointer externalType() const override { return shared_from_this(); }
|
||||||
|
|
||||||
int getNumBytes() const { return m_bytes; }
|
int getNumBytes() const { return m_bytes; }
|
||||||
|
|
||||||
@ -311,12 +318,13 @@ public:
|
|||||||
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
virtual TypePointer unaryOperatorResult(Token::Value _operator) const override;
|
||||||
virtual TypePointer binaryOperatorResult(Token::Value _operator, TypePointer const& _other) const override;
|
virtual TypePointer binaryOperatorResult(Token::Value _operator, TypePointer const& _other) const override;
|
||||||
|
|
||||||
virtual unsigned getCalldataEncodedSize(bool _padded) const { return _padded ? 32 : 1; }
|
virtual unsigned getCalldataEncodedSize(bool _padded) const override{ return _padded ? 32 : 1; }
|
||||||
virtual unsigned getStorageBytes() const override { return 1; }
|
virtual unsigned getStorageBytes() const override { return 1; }
|
||||||
virtual bool isValueType() const override { return true; }
|
virtual bool isValueType() const override { return true; }
|
||||||
|
|
||||||
virtual std::string toString() const override { return "bool"; }
|
virtual std::string toString() const override { return "bool"; }
|
||||||
virtual u256 literalValue(Literal const* _literal) const override;
|
virtual u256 literalValue(Literal const* _literal) const override;
|
||||||
|
virtual TypePointer externalType() const override { return shared_from_this(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -361,6 +369,7 @@ public:
|
|||||||
virtual unsigned getSizeOnStack() const override;
|
virtual unsigned getSizeOnStack() const override;
|
||||||
virtual std::string toString() const override;
|
virtual std::string toString() const override;
|
||||||
virtual MemberList const& getMembers() const override { return s_arrayTypeMemberList; }
|
virtual MemberList const& getMembers() const override { return s_arrayTypeMemberList; }
|
||||||
|
virtual TypePointer externalType() const override;
|
||||||
|
|
||||||
Location getLocation() const { return m_location; }
|
Location getLocation() const { return m_location; }
|
||||||
bool isByteArray() const { return m_isByteArray; }
|
bool isByteArray() const { return m_isByteArray; }
|
||||||
@ -400,6 +409,7 @@ public:
|
|||||||
virtual std::string toString() const override;
|
virtual std::string toString() const override;
|
||||||
|
|
||||||
virtual MemberList const& getMembers() const override;
|
virtual MemberList const& getMembers() const override;
|
||||||
|
virtual TypePointer externalType() const override { return std::make_shared<IntegerType>(160, IntegerType::Modifier::Address); }
|
||||||
|
|
||||||
bool isSuper() const { return m_super; }
|
bool isSuper() const { return m_super; }
|
||||||
ContractDefinition const& getContractDefinition() const { return m_contract; }
|
ContractDefinition const& getContractDefinition() const { return m_contract; }
|
||||||
@ -468,6 +478,7 @@ public:
|
|||||||
virtual bool isValueType() const override { return true; }
|
virtual bool isValueType() const override { return true; }
|
||||||
|
|
||||||
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
||||||
|
virtual TypePointer externalType() const override { return std::make_shared<IntegerType>(8 * int(getStorageBytes())); }
|
||||||
|
|
||||||
EnumDefinition const& getEnumDefinition() const { return m_enum; }
|
EnumDefinition const& getEnumDefinition() const { return m_enum; }
|
||||||
/// @returns the value that the string has in the Enum
|
/// @returns the value that the string has in the Enum
|
||||||
@ -500,6 +511,9 @@ public:
|
|||||||
Bare };
|
Bare };
|
||||||
|
|
||||||
virtual Category getCategory() const override { return Category::Function; }
|
virtual Category getCategory() const override { return Category::Function; }
|
||||||
|
|
||||||
|
virtual TypePointer externalType() const override;
|
||||||
|
|
||||||
explicit FunctionType(FunctionDefinition const& _function, bool _isInternal = true);
|
explicit FunctionType(FunctionDefinition const& _function, bool _isInternal = true);
|
||||||
explicit FunctionType(VariableDeclaration const& _varDecl);
|
explicit FunctionType(VariableDeclaration const& _varDecl);
|
||||||
explicit FunctionType(EventDefinition const& _event);
|
explicit FunctionType(EventDefinition const& _event);
|
||||||
@ -539,10 +553,10 @@ public:
|
|||||||
virtual MemberList const& getMembers() const override;
|
virtual MemberList const& getMembers() const override;
|
||||||
|
|
||||||
Location const& getLocation() const { return m_location; }
|
Location const& getLocation() const { return m_location; }
|
||||||
/// @returns the canonical signature of this function type given the function name
|
/// @returns the external signature of this function type given the function name
|
||||||
/// If @a _name is not provided (empty string) then the @c m_declaration member of the
|
/// If @a _name is not provided (empty string) then the @c m_declaration member of the
|
||||||
/// function type is used
|
/// function type is used
|
||||||
std::string getCanonicalSignature(std::string const& _name = "") const;
|
std::string externalSignature(std::string const& _name = "") const;
|
||||||
Declaration const& getDeclaration() const
|
Declaration const& getDeclaration() const
|
||||||
{
|
{
|
||||||
solAssert(m_declaration, "Requested declaration from a FunctionType that has none");
|
solAssert(m_declaration, "Requested declaration from a FunctionType that has none");
|
||||||
|
Loading…
Reference in New Issue
Block a user