mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #768 from chriseth/sol_contractsAreAddresses
Contracts inherit all address members
This commit is contained in:
commit
94cff9684f
@ -194,13 +194,7 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
|
|||||||
solAssert(_functionCall.getArguments().size() == 1, "");
|
solAssert(_functionCall.getArguments().size() == 1, "");
|
||||||
Expression const& firstArgument = *_functionCall.getArguments().front();
|
Expression const& firstArgument = *_functionCall.getArguments().front();
|
||||||
firstArgument.accept(*this);
|
firstArgument.accept(*this);
|
||||||
if (firstArgument.getType()->getCategory() == Type::Category::CONTRACT &&
|
appendTypeConversion(*firstArgument.getType(), *_functionCall.getType());
|
||||||
_functionCall.getType()->getCategory() == Type::Category::INTEGER)
|
|
||||||
{
|
|
||||||
// explicit type conversion contract -> address, nothing to do.
|
|
||||||
}
|
|
||||||
else
|
|
||||||
appendTypeConversion(*firstArgument.getType(), *_functionCall.getType());
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -250,13 +244,17 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
|
|||||||
FunctionCallOptions options;
|
FunctionCallOptions options;
|
||||||
options.bare = true;
|
options.bare = true;
|
||||||
options.obtainAddress = [&]() { _functionCall.getExpression().accept(*this); };
|
options.obtainAddress = [&]() { _functionCall.getExpression().accept(*this); };
|
||||||
options.obtainValue = [&]() { arguments.front()->accept(*this); };
|
options.obtainValue = [&]()
|
||||||
|
{
|
||||||
|
arguments.front()->accept(*this);
|
||||||
|
appendTypeConversion(*arguments.front()->getType(),
|
||||||
|
*function.getParameterTypes().front(), true);
|
||||||
|
};
|
||||||
appendExternalFunctionCall(FunctionType({}, {}, Location::EXTERNAL), {}, options);
|
appendExternalFunctionCall(FunctionType({}, {}, Location::EXTERNAL), {}, options);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case Location::SUICIDE:
|
case Location::SUICIDE:
|
||||||
arguments.front()->accept(*this);
|
arguments.front()->accept(*this);
|
||||||
//@todo might not be necessary
|
|
||||||
appendTypeConversion(*arguments.front()->getType(), *function.getParameterTypes().front(), true);
|
appendTypeConversion(*arguments.front()->getType(), *function.getParameterTypes().front(), true);
|
||||||
m_context << eth::Instruction::SUICIDE;
|
m_context << eth::Instruction::SUICIDE;
|
||||||
break;
|
break;
|
||||||
@ -347,6 +345,18 @@ void ExpressionCompiler::endVisit(MemberAccess const& _memberAccess)
|
|||||||
ASTString const& member = _memberAccess.getMemberName();
|
ASTString const& member = _memberAccess.getMemberName();
|
||||||
switch (_memberAccess.getExpression().getType()->getCategory())
|
switch (_memberAccess.getExpression().getType()->getCategory())
|
||||||
{
|
{
|
||||||
|
case Type::Category::CONTRACT:
|
||||||
|
{
|
||||||
|
ContractType const& type = dynamic_cast<ContractType const&>(*_memberAccess.getExpression().getType());
|
||||||
|
u256 identifier = type.getFunctionIdentifier(member);
|
||||||
|
if (identifier != Invalid256)
|
||||||
|
{
|
||||||
|
appendTypeConversion(type, IntegerType(0, IntegerType::Modifier::ADDRESS), true);
|
||||||
|
m_context << identifier;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// fall-through to "integer" otherwise (address)
|
||||||
|
}
|
||||||
case Type::Category::INTEGER:
|
case Type::Category::INTEGER:
|
||||||
if (member == "balance")
|
if (member == "balance")
|
||||||
{
|
{
|
||||||
@ -360,12 +370,6 @@ void ExpressionCompiler::endVisit(MemberAccess const& _memberAccess)
|
|||||||
else
|
else
|
||||||
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("Invalid member access to integer."));
|
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("Invalid member access to integer."));
|
||||||
break;
|
break;
|
||||||
case Type::Category::CONTRACT:
|
|
||||||
{
|
|
||||||
ContractType const& type = dynamic_cast<ContractType const&>(*_memberAccess.getExpression().getType());
|
|
||||||
m_context << type.getFunctionIdentifier(member);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case Type::Category::MAGIC:
|
case Type::Category::MAGIC:
|
||||||
// we can ignore the kind of magic and only look at the name of the member
|
// we can ignore the kind of magic and only look at the name of the member
|
||||||
if (member == "coinbase")
|
if (member == "coinbase")
|
||||||
@ -592,15 +596,36 @@ void ExpressionCompiler::appendTypeConversion(Type const& _typeOnStack, Type con
|
|||||||
return;
|
return;
|
||||||
Type::Category stackTypeCategory = _typeOnStack.getCategory();
|
Type::Category stackTypeCategory = _typeOnStack.getCategory();
|
||||||
Type::Category targetTypeCategory = _targetType.getCategory();
|
Type::Category targetTypeCategory = _targetType.getCategory();
|
||||||
if (stackTypeCategory == Type::Category::INTEGER)
|
if (stackTypeCategory == Type::Category::INTEGER || stackTypeCategory == Type::Category::CONTRACT ||
|
||||||
|
stackTypeCategory == Type::Category::INTEGER_CONSTANT)
|
||||||
{
|
{
|
||||||
solAssert(targetTypeCategory == Type::Category::INTEGER || targetTypeCategory == Type::Category::CONTRACT, "");
|
solAssert(targetTypeCategory == Type::Category::INTEGER || targetTypeCategory == Type::Category::CONTRACT, "");
|
||||||
appendHighBitsCleanup(dynamic_cast<IntegerType const&>(_typeOnStack));
|
IntegerType addressType(0, IntegerType::Modifier::ADDRESS);
|
||||||
|
IntegerType const& targetType = targetTypeCategory == Type::Category::INTEGER
|
||||||
|
? dynamic_cast<IntegerType const&>(_targetType) : addressType;
|
||||||
|
if (stackTypeCategory == Type::Category::INTEGER_CONSTANT)
|
||||||
|
{
|
||||||
|
IntegerConstantType const& constType = dynamic_cast<IntegerConstantType const&>(_typeOnStack);
|
||||||
|
// We know that the stack is clean, we only have to clean for a narrowing conversion
|
||||||
|
// where cleanup is forced.
|
||||||
|
if (targetType.getNumBits() < constType.getIntegerType()->getNumBits() && _cleanupNeeded)
|
||||||
|
appendHighBitsCleanup(targetType);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
IntegerType const& typeOnStack = stackTypeCategory == Type::Category::INTEGER
|
||||||
|
? dynamic_cast<IntegerType const&>(_typeOnStack) : addressType;
|
||||||
|
// Widening: clean up according to source type width
|
||||||
|
// Non-widening and force: clean up according to target type bits
|
||||||
|
if (targetType.getNumBits() > typeOnStack.getNumBits())
|
||||||
|
appendHighBitsCleanup(typeOnStack);
|
||||||
|
else if (_cleanupNeeded)
|
||||||
|
appendHighBitsCleanup(targetType);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (stackTypeCategory == Type::Category::INTEGER_CONSTANT)
|
|
||||||
solAssert(targetTypeCategory == Type::Category::INTEGER || targetTypeCategory == Type::Category::CONTRACT, "");
|
|
||||||
else if (stackTypeCategory == Type::Category::STRING)
|
else if (stackTypeCategory == Type::Category::STRING)
|
||||||
{
|
{
|
||||||
|
solAssert(targetTypeCategory == Type::Category::STRING, "");
|
||||||
// nothing to do, strings are high-order-bit-aligned
|
// nothing to do, strings are high-order-bit-aligned
|
||||||
//@todo clear lower-order bytes if we allow explicit conversion to shorter strings
|
//@todo clear lower-order bytes if we allow explicit conversion to shorter strings
|
||||||
}
|
}
|
||||||
|
15
Types.cpp
15
Types.cpp
@ -424,15 +424,20 @@ TypePointer BoolType::binaryOperatorResult(Token::Value _operator, TypePointer c
|
|||||||
return TypePointer();
|
return TypePointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ContractType::isExplicitlyConvertibleTo(Type const& _convertTo) const
|
bool ContractType::isImplicitlyConvertibleTo(Type const& _convertTo) const
|
||||||
{
|
{
|
||||||
if (isImplicitlyConvertibleTo(_convertTo))
|
if (*this == _convertTo)
|
||||||
return true;
|
return true;
|
||||||
if (_convertTo.getCategory() == Category::INTEGER)
|
if (_convertTo.getCategory() == Category::INTEGER)
|
||||||
return dynamic_cast<IntegerType const&>(_convertTo).isAddress();
|
return dynamic_cast<IntegerType const&>(_convertTo).isAddress();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ContractType::isExplicitlyConvertibleTo(Type const& _convertTo) const
|
||||||
|
{
|
||||||
|
return isImplicitlyConvertibleTo(_convertTo) || _convertTo.getCategory() == Category::INTEGER;
|
||||||
|
}
|
||||||
|
|
||||||
bool ContractType::operator==(Type const& _other) const
|
bool ContractType::operator==(Type const& _other) const
|
||||||
{
|
{
|
||||||
if (_other.getCategory() != getCategory())
|
if (_other.getCategory() != getCategory())
|
||||||
@ -459,7 +464,9 @@ MemberList const& ContractType::getMembers() const
|
|||||||
// We need to lazy-initialize it because of recursive references.
|
// We need to lazy-initialize it because of recursive references.
|
||||||
if (!m_members)
|
if (!m_members)
|
||||||
{
|
{
|
||||||
map<string, shared_ptr<Type const>> members;
|
// All address members and all interface functions
|
||||||
|
map<string, shared_ptr<Type const>> members(IntegerType::AddressMemberList.begin(),
|
||||||
|
IntegerType::AddressMemberList.end());
|
||||||
for (auto const& it: m_contract.getInterfaceFunctions())
|
for (auto const& it: m_contract.getInterfaceFunctions())
|
||||||
members[it.second->getName()] = make_shared<FunctionType>(*it.second, false);
|
members[it.second->getName()] = make_shared<FunctionType>(*it.second, false);
|
||||||
m_members.reset(new MemberList(members));
|
m_members.reset(new MemberList(members));
|
||||||
@ -487,7 +494,7 @@ u256 ContractType::getFunctionIdentifier(string const& _functionName) const
|
|||||||
if (it->second->getName() == _functionName)
|
if (it->second->getName() == _functionName)
|
||||||
return FixedHash<4>::Arith(it->first);
|
return FixedHash<4>::Arith(it->first);
|
||||||
|
|
||||||
BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("Index of non-existing contract function requested."));
|
return Invalid256;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool StructType::operator==(Type const& _other) const
|
bool StructType::operator==(Type const& _other) const
|
||||||
|
9
Types.h
9
Types.h
@ -180,10 +180,11 @@ public:
|
|||||||
bool isAddress() const { return m_modifier == Modifier::ADDRESS; }
|
bool isAddress() const { return m_modifier == Modifier::ADDRESS; }
|
||||||
int isSigned() const { return m_modifier == Modifier::SIGNED; }
|
int isSigned() const { return m_modifier == Modifier::SIGNED; }
|
||||||
|
|
||||||
|
static const MemberList AddressMemberList;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int m_bits;
|
int m_bits;
|
||||||
Modifier m_modifier;
|
Modifier m_modifier;
|
||||||
static const MemberList AddressMemberList;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -279,7 +280,9 @@ class ContractType: public Type
|
|||||||
public:
|
public:
|
||||||
virtual Category getCategory() const override { return Category::CONTRACT; }
|
virtual Category getCategory() const override { return Category::CONTRACT; }
|
||||||
ContractType(ContractDefinition const& _contract): m_contract(_contract) {}
|
ContractType(ContractDefinition const& _contract): m_contract(_contract) {}
|
||||||
/// Contracts can be converted to themselves and to addresses.
|
/// Contracts can be implicitly converted to super classes and to addresses.
|
||||||
|
virtual bool isImplicitlyConvertibleTo(Type const& _convertTo) const override;
|
||||||
|
/// Contracts can be converted to themselves and to integers.
|
||||||
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
virtual bool isExplicitlyConvertibleTo(Type const& _convertTo) const override;
|
||||||
virtual bool operator==(Type const& _other) const override;
|
virtual bool operator==(Type const& _other) const override;
|
||||||
virtual u256 getStorageSize() const override;
|
virtual u256 getStorageSize() const override;
|
||||||
@ -292,6 +295,8 @@ public:
|
|||||||
/// is not used, as this type cannot be the type of a variable or expression.
|
/// is not used, as this type cannot be the type of a variable or expression.
|
||||||
std::shared_ptr<FunctionType const> const& getConstructorType() const;
|
std::shared_ptr<FunctionType const> const& getConstructorType() const;
|
||||||
|
|
||||||
|
/// @returns the identifier of the function with the given name or Invalid256 if such a name does
|
||||||
|
/// not exist.
|
||||||
u256 getFunctionIdentifier(std::string const& _functionName) const;
|
u256 getFunctionIdentifier(std::string const& _functionName) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
Loading…
Reference in New Issue
Block a user