mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Introduced tuple type and added multi variable declarations to type
checker.
This commit is contained in:
parent
c54a033bf0
commit
a5d12b8761
@ -558,7 +558,9 @@ protected:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
ASTPointer<TypeName> m_typeName; ///< can be empty ("var")
|
ASTPointer<TypeName> m_typeName; ///< can be empty ("var")
|
||||||
ASTPointer<Expression> m_value; ///< the assigned value, can be missing
|
/// Initially assigned value, can be missing. For local variables, this is stored inside
|
||||||
|
/// VariableDeclarationStatement and not here.
|
||||||
|
ASTPointer<Expression> m_value;
|
||||||
bool m_isStateVariable; ///< Whether or not this is a contract state variable
|
bool m_isStateVariable; ///< Whether or not this is a contract state variable
|
||||||
bool m_isIndexed; ///< Whether this is an indexed variable (used by events).
|
bool m_isIndexed; ///< Whether this is an indexed variable (used by events).
|
||||||
bool m_isConstant; ///< Whether the variable is a compile-time constant.
|
bool m_isConstant; ///< Whether the variable is a compile-time constant.
|
||||||
|
@ -783,14 +783,34 @@ ASTPointer<VariableDeclarationStatement> Parser::parseVariableDeclarationStateme
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
// Parse `var (a, b, ,, c) = ...` into a single VariableDeclarationStatement with multiple variables.
|
// Parse `var (a, b, ,, c) = ...` into a single VariableDeclarationStatement with multiple variables.
|
||||||
solAssert(false, "To be implemented.");
|
m_scanner->next();
|
||||||
|
m_scanner->next();
|
||||||
|
do
|
||||||
|
{
|
||||||
|
ASTPointer<VariableDeclaration> var;
|
||||||
|
if (m_scanner->currentToken() == Token::Comma)
|
||||||
|
m_scanner->next();
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ASTNodeFactory varDeclNodeFactory(*this);
|
||||||
|
ASTPointer<ASTString> name = expectIdentifierToken();
|
||||||
|
var = varDeclNodeFactory.createNode<VariableDeclaration>(
|
||||||
|
ASTPointer<TypeName>(),
|
||||||
|
name,
|
||||||
|
ASTPointer<Expression>(),
|
||||||
|
VariableDeclaration::Visibility::Default
|
||||||
|
);
|
||||||
|
}
|
||||||
|
variables.push_back(var);
|
||||||
|
} while (m_scanner->currentToken() != Token::RParen);
|
||||||
|
nodeFactory.markEndPosition();
|
||||||
|
m_scanner->next();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
VarDeclParserOptions options;
|
VarDeclParserOptions options;
|
||||||
options.allowVar = true;
|
options.allowVar = true;
|
||||||
options.allowLocationSpecifier = true;
|
options.allowLocationSpecifier = true;
|
||||||
options.allowInitialValue = false;
|
|
||||||
variables.push_back(parseVariableDeclaration(options, _lookAheadArrayType));
|
variables.push_back(parseVariableDeclaration(options, _lookAheadArrayType));
|
||||||
}
|
}
|
||||||
if (m_scanner->currentToken() == Token::Assign)
|
if (m_scanner->currentToken() == Token::Assign)
|
||||||
|
@ -449,18 +449,6 @@ bool TypeChecker::visit(VariableDeclaration const& _variable)
|
|||||||
}
|
}
|
||||||
if (_variable.value())
|
if (_variable.value())
|
||||||
expectType(*_variable.value(), *varType);
|
expectType(*_variable.value(), *varType);
|
||||||
else
|
|
||||||
{
|
|
||||||
if (auto ref = dynamic_cast<ReferenceType const *>(varType.get()))
|
|
||||||
if (ref->dataStoredIn(DataLocation::Storage) && _variable.isLocalVariable() && !_variable.isCallableParameter())
|
|
||||||
{
|
|
||||||
auto err = make_shared<Warning>();
|
|
||||||
*err <<
|
|
||||||
errinfo_sourceLocation(_variable.location()) <<
|
|
||||||
errinfo_comment("Uninitialized storage pointer. Did you mean '<type> memory " + _variable.name() + "'?");
|
|
||||||
m_errors.push_back(err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!_variable.isStateVariable())
|
if (!_variable.isStateVariable())
|
||||||
{
|
{
|
||||||
if (varType->dataStoredIn(DataLocation::Memory) || varType->dataStoredIn(DataLocation::CallData))
|
if (varType->dataStoredIn(DataLocation::Memory) || varType->dataStoredIn(DataLocation::CallData))
|
||||||
@ -601,36 +589,101 @@ void TypeChecker::endVisit(Return const& _return)
|
|||||||
|
|
||||||
bool TypeChecker::visit(VariableDeclarationStatement const& _statement)
|
bool TypeChecker::visit(VariableDeclarationStatement const& _statement)
|
||||||
{
|
{
|
||||||
solAssert(_statement.declarations().size() == 1, "To be implemented.");
|
solAssert(!_statement.declarations().empty(), "");
|
||||||
solAssert(!!_statement.declarations().front(), "");
|
if (!_statement.initialValue())
|
||||||
VariableDeclaration const& var = *_statement.declarations().front();
|
{
|
||||||
|
// No initial value is only permitted for single variables with specified type.
|
||||||
|
if (_statement.declarations().size() != 1 || !_statement.declarations().front())
|
||||||
|
fatalTypeError(_statement, "Assignment necessary for type detection.");
|
||||||
|
VariableDeclaration const& varDecl = *_statement.declarations().front();
|
||||||
|
if (!varDecl.annotation().type)
|
||||||
|
fatalTypeError(_statement, "Assignment necessary for type detection.");
|
||||||
|
if (auto ref = dynamic_cast<ReferenceType const*>(varDecl.annotation().type.get()))
|
||||||
|
{
|
||||||
|
if (ref->dataStoredIn(DataLocation::Storage))
|
||||||
|
{
|
||||||
|
auto err = make_shared<Warning>();
|
||||||
|
*err <<
|
||||||
|
errinfo_sourceLocation(varDecl.location()) <<
|
||||||
|
errinfo_comment("Uninitialized storage pointer. Did you mean '<type> memory " + varDecl.name() + "'?");
|
||||||
|
m_errors.push_back(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
varDecl.accept(*this);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Here we have an initial value and might have to derive some types before we can visit
|
||||||
|
// the variable declaration(s).
|
||||||
|
|
||||||
|
_statement.initialValue()->accept(*this);
|
||||||
|
shared_ptr<TupleType const> valueType = dynamic_pointer_cast<TupleType const>(_statement.initialValue()->annotation().type);
|
||||||
|
if (!valueType)
|
||||||
|
valueType = make_shared<TupleType const>(TypePointers{_statement.initialValue()->annotation().type});
|
||||||
|
|
||||||
|
vector<ASTPointer<VariableDeclaration>> variables = _statement.declarations();
|
||||||
|
// If numbers do not match, fill up if variables begin or end empty (not both).
|
||||||
|
if (valueType->components().size() != variables.size())
|
||||||
|
{
|
||||||
|
if (!variables.front() && !variables.back())
|
||||||
|
fatalTypeError(
|
||||||
|
_statement,
|
||||||
|
"Wildcard both at beginning and end of variable declaration list is only allowed "
|
||||||
|
"if the number of components is equal."
|
||||||
|
);
|
||||||
|
while (valueType->components().size() > variables.size())
|
||||||
|
if (!variables.front())
|
||||||
|
variables.insert(variables.begin(), shared_ptr<VariableDeclaration>());
|
||||||
|
else
|
||||||
|
variables.push_back(shared_ptr<VariableDeclaration>());
|
||||||
|
while (valueType->components().size() < variables.size())
|
||||||
|
if (!variables.empty() && !variables.front())
|
||||||
|
variables.erase(variables.begin());
|
||||||
|
else if (!variables.empty() && !variables.back())
|
||||||
|
variables.pop_back();
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
if (valueType->components().size() != variables.size())
|
||||||
|
fatalTypeError(
|
||||||
|
_statement,
|
||||||
|
"Unable to match the number of variables to the number of values."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
solAssert(variables.size() == valueType->components().size(), "");
|
||||||
|
|
||||||
|
for (size_t i = 0; i < variables.size(); ++i)
|
||||||
|
{
|
||||||
|
if (!variables[i])
|
||||||
|
continue;
|
||||||
|
VariableDeclaration const& var = *variables[i];
|
||||||
solAssert(!var.value(), "Value has to be tied to statement.");
|
solAssert(!var.value(), "Value has to be tied to statement.");
|
||||||
|
TypePointer const& valueComponentType = valueType->components()[i];
|
||||||
|
solAssert(!!valueComponentType, "");
|
||||||
if (!var.annotation().type)
|
if (!var.annotation().type)
|
||||||
{
|
{
|
||||||
solAssert(!var.typeName(), "");
|
|
||||||
// Infer type from value.
|
// Infer type from value.
|
||||||
if (!_statement.initialValue())
|
solAssert(!var.typeName(), "");
|
||||||
fatalTypeError(_statement, "Assignment necessary for type detection.");
|
|
||||||
_statement.initialValue()->accept(*this);
|
|
||||||
|
|
||||||
TypePointer const& valueType = type(*_statement.initialValue());
|
|
||||||
solAssert(!!valueType, "");
|
|
||||||
if (
|
if (
|
||||||
valueType->category() == Type::Category::IntegerConstant &&
|
valueComponentType->category() == Type::Category::IntegerConstant &&
|
||||||
!dynamic_pointer_cast<IntegerConstantType const>(valueType)->integerType()
|
!dynamic_pointer_cast<IntegerConstantType const>(valueComponentType)->integerType()
|
||||||
)
|
)
|
||||||
fatalTypeError(*_statement.initialValue(), "Invalid integer constant " + valueType->toString() + ".");
|
fatalTypeError(*_statement.initialValue(), "Invalid integer constant " + valueComponentType->toString() + ".");
|
||||||
else if (valueType->category() == Type::Category::Void)
|
var.annotation().type = valueComponentType->mobileType();
|
||||||
fatalTypeError(_statement, "Variable cannot have void type.");
|
|
||||||
var.annotation().type = valueType->mobileType();
|
|
||||||
var.accept(*this);
|
var.accept(*this);
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var.accept(*this);
|
var.accept(*this);
|
||||||
if (_statement.initialValue())
|
if (!valueComponentType->isImplicitlyConvertibleTo(*var.annotation().type))
|
||||||
expectType(*_statement.initialValue(), *var.annotation().type);
|
typeError(
|
||||||
|
_statement,
|
||||||
|
"Type " +
|
||||||
|
valueComponentType->toString() +
|
||||||
|
" is not implicitly convertible to expected type " +
|
||||||
|
var.annotation().type->toString() +
|
||||||
|
"."
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -799,23 +852,14 @@ bool TypeChecker::visit(FunctionCall const& _functionCall)
|
|||||||
if (!functionType)
|
if (!functionType)
|
||||||
{
|
{
|
||||||
typeError(_functionCall, "Type is not callable");
|
typeError(_functionCall, "Type is not callable");
|
||||||
_functionCall.annotation().type = make_shared<VoidType>();
|
_functionCall.annotation().type = make_shared<TupleType>();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
else
|
else if (functionType->returnParameterTypes().size() == 1)
|
||||||
{
|
|
||||||
// @todo actually the return type should be an anonymous struct,
|
|
||||||
// but we change it to the type of the first return value until we have anonymous
|
|
||||||
// structs and tuples
|
|
||||||
if (functionType->returnParameterTypes().empty())
|
|
||||||
_functionCall.annotation().type = make_shared<VoidType>();
|
|
||||||
else
|
|
||||||
_functionCall.annotation().type = functionType->returnParameterTypes().front();
|
_functionCall.annotation().type = functionType->returnParameterTypes().front();
|
||||||
}
|
else
|
||||||
|
_functionCall.annotation().type = make_shared<TupleType>(functionType->returnParameterTypes());
|
||||||
|
|
||||||
//@todo would be nice to create a struct type from the arguments
|
|
||||||
// and then ask if that is implicitly convertible to the struct represented by the
|
|
||||||
// function parameters
|
|
||||||
TypePointers const& parameterTypes = functionType->parameterTypes();
|
TypePointers const& parameterTypes = functionType->parameterTypes();
|
||||||
if (!functionType->takesArbitraryParameters() && parameterTypes.size() != arguments.size())
|
if (!functionType->takesArbitraryParameters() && parameterTypes.size() != arguments.size())
|
||||||
{
|
{
|
||||||
|
@ -223,7 +223,7 @@ TypePointer IntegerType::unaryOperatorResult(Token::Value _operator) const
|
|||||||
{
|
{
|
||||||
// "delete" is ok for all integer types
|
// "delete" is ok for all integer types
|
||||||
if (_operator == Token::Delete)
|
if (_operator == Token::Delete)
|
||||||
return make_shared<VoidType>();
|
return make_shared<TupleType>();
|
||||||
// no further unary operators for addresses
|
// no further unary operators for addresses
|
||||||
else if (isAddress())
|
else if (isAddress())
|
||||||
return TypePointer();
|
return TypePointer();
|
||||||
@ -562,7 +562,7 @@ TypePointer FixedBytesType::unaryOperatorResult(Token::Value _operator) const
|
|||||||
{
|
{
|
||||||
// "delete" and "~" is okay for FixedBytesType
|
// "delete" and "~" is okay for FixedBytesType
|
||||||
if (_operator == Token::Delete)
|
if (_operator == Token::Delete)
|
||||||
return make_shared<VoidType>();
|
return make_shared<TupleType>();
|
||||||
else if (_operator == Token::BitNot)
|
else if (_operator == Token::BitNot)
|
||||||
return shared_from_this();
|
return shared_from_this();
|
||||||
|
|
||||||
@ -617,7 +617,7 @@ u256 BoolType::literalValue(Literal const* _literal) const
|
|||||||
TypePointer BoolType::unaryOperatorResult(Token::Value _operator) const
|
TypePointer BoolType::unaryOperatorResult(Token::Value _operator) const
|
||||||
{
|
{
|
||||||
if (_operator == Token::Delete)
|
if (_operator == Token::Delete)
|
||||||
return make_shared<VoidType>();
|
return make_shared<TupleType>();
|
||||||
return (_operator == Token::Not) ? shared_from_this() : TypePointer();
|
return (_operator == Token::Not) ? shared_from_this() : TypePointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -658,7 +658,7 @@ bool ContractType::isExplicitlyConvertibleTo(Type const& _convertTo) const
|
|||||||
|
|
||||||
TypePointer ContractType::unaryOperatorResult(Token::Value _operator) const
|
TypePointer ContractType::unaryOperatorResult(Token::Value _operator) const
|
||||||
{
|
{
|
||||||
return _operator == Token::Delete ? make_shared<VoidType>() : TypePointer();
|
return _operator == Token::Delete ? make_shared<TupleType>() : TypePointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePointer ReferenceType::unaryOperatorResult(Token::Value _operator) const
|
TypePointer ReferenceType::unaryOperatorResult(Token::Value _operator) const
|
||||||
@ -672,9 +672,9 @@ TypePointer ReferenceType::unaryOperatorResult(Token::Value _operator) const
|
|||||||
case DataLocation::CallData:
|
case DataLocation::CallData:
|
||||||
return TypePointer();
|
return TypePointer();
|
||||||
case DataLocation::Memory:
|
case DataLocation::Memory:
|
||||||
return make_shared<VoidType>();
|
return make_shared<TupleType>();
|
||||||
case DataLocation::Storage:
|
case DataLocation::Storage:
|
||||||
return m_isPointer ? TypePointer() : make_shared<VoidType>();
|
return m_isPointer ? TypePointer() : make_shared<TupleType>();
|
||||||
default:
|
default:
|
||||||
solAssert(false, "");
|
solAssert(false, "");
|
||||||
}
|
}
|
||||||
@ -1175,7 +1175,7 @@ set<string> StructType::membersMissingInMemory() const
|
|||||||
|
|
||||||
TypePointer EnumType::unaryOperatorResult(Token::Value _operator) const
|
TypePointer EnumType::unaryOperatorResult(Token::Value _operator) const
|
||||||
{
|
{
|
||||||
return _operator == Token::Delete ? make_shared<VoidType>() : TypePointer();
|
return _operator == Token::Delete ? make_shared<TupleType>() : TypePointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool EnumType::operator==(Type const& _other) const
|
bool EnumType::operator==(Type const& _other) const
|
||||||
@ -1222,6 +1222,40 @@ unsigned int EnumType::memberValue(ASTString const& _member) const
|
|||||||
BOOST_THROW_EXCEPTION(m_enum.createTypeError("Requested unknown enum value ." + _member));
|
BOOST_THROW_EXCEPTION(m_enum.createTypeError("Requested unknown enum value ." + _member));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool TupleType::operator==(Type const& _other) const
|
||||||
|
{
|
||||||
|
if (auto tupleType = dynamic_cast<TupleType const*>(&_other))
|
||||||
|
return components() == tupleType->components();
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
string TupleType::toString(bool _short) const
|
||||||
|
{
|
||||||
|
if (m_components.empty())
|
||||||
|
return "tuple()";
|
||||||
|
string str = "tuple(";
|
||||||
|
for (auto const& t: m_components)
|
||||||
|
str += t->toString(_short) + ", ";
|
||||||
|
str.resize(str.size() - 2);
|
||||||
|
return str + ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
u256 TupleType::storageSize() const
|
||||||
|
{
|
||||||
|
BOOST_THROW_EXCEPTION(
|
||||||
|
InternalCompilerError()
|
||||||
|
<< errinfo_comment("Storage size of non-storable tuple type requested."));
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned TupleType::sizeOnStack() const
|
||||||
|
{
|
||||||
|
unsigned size = 0;
|
||||||
|
for (auto const& t: m_components)
|
||||||
|
size += t->sizeOnStack();
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
FunctionType::FunctionType(FunctionDefinition const& _function, bool _isInternal):
|
FunctionType::FunctionType(FunctionDefinition const& _function, bool _isInternal):
|
||||||
m_location(_isInternal ? Location::Internal : Location::External),
|
m_location(_isInternal ? Location::Internal : Location::External),
|
||||||
m_isConstant(_function.isDeclaredConst()),
|
m_isConstant(_function.isDeclaredConst()),
|
||||||
@ -1647,13 +1681,6 @@ string MappingType::canonicalName(bool) const
|
|||||||
return "mapping(" + keyType()->canonicalName(false) + " => " + valueType()->canonicalName(false) + ")";
|
return "mapping(" + keyType()->canonicalName(false) + " => " + valueType()->canonicalName(false) + ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
u256 VoidType::storageSize() const
|
|
||||||
{
|
|
||||||
BOOST_THROW_EXCEPTION(
|
|
||||||
InternalCompilerError()
|
|
||||||
<< errinfo_comment("Storage size of non-storable void type requested."));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TypeType::operator==(Type const& _other) const
|
bool TypeType::operator==(Type const& _other) const
|
||||||
{
|
{
|
||||||
if (_other.category() != category())
|
if (_other.category() != category())
|
||||||
|
@ -132,8 +132,8 @@ public:
|
|||||||
enum class Category
|
enum class Category
|
||||||
{
|
{
|
||||||
Integer, IntegerConstant, StringLiteral, Bool, Real, Array,
|
Integer, IntegerConstant, StringLiteral, Bool, Real, Array,
|
||||||
FixedBytes, Contract, Struct, Function, Enum,
|
FixedBytes, Contract, Struct, Function, Enum, Tuple,
|
||||||
Mapping, Void, TypeType, Modifier, Magic
|
Mapping, TypeType, Modifier, Magic
|
||||||
};
|
};
|
||||||
|
|
||||||
/// @{
|
/// @{
|
||||||
@ -682,6 +682,28 @@ private:
|
|||||||
mutable std::unique_ptr<MemberList> m_members;
|
mutable std::unique_ptr<MemberList> m_members;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Type that can hold a finite sequence of values of different types.
|
||||||
|
*/
|
||||||
|
class TupleType: public Type
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual Category category() const override { return Category::Tuple; }
|
||||||
|
explicit TupleType(std::vector<TypePointer> const& _types = std::vector<TypePointer>()): m_components(_types) {}
|
||||||
|
virtual bool operator==(Type const& _other) const override;
|
||||||
|
virtual TypePointer binaryOperatorResult(Token::Value, TypePointer const&) const override { return TypePointer(); }
|
||||||
|
virtual std::string toString(bool) const override;
|
||||||
|
virtual bool canBeStored() const override { return false; }
|
||||||
|
virtual u256 storageSize() const override;
|
||||||
|
virtual bool canLiveOutsideStorage() const override { return false; }
|
||||||
|
virtual unsigned sizeOnStack() const override;
|
||||||
|
|
||||||
|
std::vector<TypePointer> const& components() const { return m_components; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<TypePointer> const m_components;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The type of a function, identified by its (return) parameter types.
|
* The type of a function, identified by its (return) parameter types.
|
||||||
* @todo the return parameters should also have names, i.e. return parameters should be a struct
|
* @todo the return parameters should also have names, i.e. return parameters should be a struct
|
||||||
@ -874,24 +896,6 @@ private:
|
|||||||
TypePointer m_valueType;
|
TypePointer m_valueType;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* The void type, can only be implicitly used as the type that is returned by functions without
|
|
||||||
* return parameters.
|
|
||||||
*/
|
|
||||||
class VoidType: public Type
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
virtual Category category() const override { return Category::Void; }
|
|
||||||
VoidType() {}
|
|
||||||
|
|
||||||
virtual TypePointer binaryOperatorResult(Token::Value, TypePointer const&) const override { return TypePointer(); }
|
|
||||||
virtual std::string toString(bool) const override { return "void"; }
|
|
||||||
virtual bool canBeStored() const override { return false; }
|
|
||||||
virtual u256 storageSize() const override;
|
|
||||||
virtual bool canLiveOutsideStorage() const override { return false; }
|
|
||||||
virtual unsigned sizeOnStack() const override { return 0; }
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The type of a type reference. The type of "uint32" when used in "a = uint32(2)" is an example
|
* The type of a type reference. The type of "uint32" when used in "a = uint32(2)" is an example
|
||||||
* of a TypeType.
|
* of a TypeType.
|
||||||
|
@ -1285,7 +1285,7 @@ BOOST_AUTO_TEST_CASE(empty_name_return_parameter_with_named_one)
|
|||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(disallow_declaration_of_void_type)
|
BOOST_AUTO_TEST_CASE(disallow_declaration_of_void_type)
|
||||||
{
|
{
|
||||||
char const* sourceCode = "contract c { function f() { var x = f(); } }";
|
char const* sourceCode = "contract c { function f() { var (x) = f(); } }";
|
||||||
SOLIDITY_CHECK_ERROR_TYPE(parseAndAnalyseReturnError(sourceCode), TypeError);
|
SOLIDITY_CHECK_ERROR_TYPE(parseAndAnalyseReturnError(sourceCode), TypeError);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2134,7 +2134,7 @@ BOOST_AUTO_TEST_CASE(dynamic_return_types_not_possible)
|
|||||||
contract C {
|
contract C {
|
||||||
function f(uint) returns (string);
|
function f(uint) returns (string);
|
||||||
function g() {
|
function g() {
|
||||||
var x = this.f(2);
|
var (x,) = this.f(2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
|
@ -952,6 +952,18 @@ BOOST_AUTO_TEST_CASE(multi_variable_declaration)
|
|||||||
BOOST_CHECK_NO_THROW(parseText(text));
|
BOOST_CHECK_NO_THROW(parseText(text));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(multi_variable_declaration_invalid)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
library Lib {
|
||||||
|
function f() {
|
||||||
|
var () = g();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
BOOST_CHECK_THROW(parseText(text), ParserError);
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_SUITE_END()
|
BOOST_AUTO_TEST_SUITE_END()
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user