mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Only active variables at the point of their declaration.
This commit is contained in:
parent
6b9dda06f3
commit
88a5c66f4a
@ -79,6 +79,17 @@ Declaration const* DeclarationContainer::conflictingDeclaration(
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DeclarationContainer::activateVariable(ASTString const& _name)
|
||||||
|
{
|
||||||
|
solAssert(
|
||||||
|
m_invisibleDeclarations.count(_name) && m_invisibleDeclarations.at(_name).size() == 1,
|
||||||
|
"Tried to activate a non-inactive variable or multiple inactive variables with the same name."
|
||||||
|
);
|
||||||
|
solAssert(m_declarations.count(_name) == 0 || m_declarations.at(_name).empty(), "");
|
||||||
|
m_declarations[_name].emplace_back(m_invisibleDeclarations.at(_name).front());
|
||||||
|
m_invisibleDeclarations.erase(_name);
|
||||||
|
}
|
||||||
|
|
||||||
bool DeclarationContainer::registerDeclaration(
|
bool DeclarationContainer::registerDeclaration(
|
||||||
Declaration const& _declaration,
|
Declaration const& _declaration,
|
||||||
ASTString const* _name,
|
ASTString const* _name,
|
||||||
@ -106,15 +117,17 @@ bool DeclarationContainer::registerDeclaration(
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<Declaration const*> DeclarationContainer::resolveName(ASTString const& _name, bool _recursive) const
|
vector<Declaration const*> DeclarationContainer::resolveName(ASTString const& _name, bool _recursive, bool _alsoInvisible) const
|
||||||
{
|
{
|
||||||
solAssert(!_name.empty(), "Attempt to resolve empty name.");
|
solAssert(!_name.empty(), "Attempt to resolve empty name.");
|
||||||
auto result = m_declarations.find(_name);
|
vector<Declaration const*> result;
|
||||||
if (result != m_declarations.end())
|
if (m_declarations.count(_name))
|
||||||
return result->second;
|
result = m_declarations.at(_name);
|
||||||
if (_recursive && m_enclosingContainer)
|
if (_alsoInvisible && m_invisibleDeclarations.count(_name))
|
||||||
return m_enclosingContainer->resolveName(_name, true);
|
result += m_invisibleDeclarations.at(_name);
|
||||||
return vector<Declaration const*>({});
|
if (result.empty() && _recursive && m_enclosingContainer)
|
||||||
|
result = m_enclosingContainer->resolveName(_name, true, _alsoInvisible);
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<ASTString> DeclarationContainer::similarNames(ASTString const& _name) const
|
vector<ASTString> DeclarationContainer::similarNames(ASTString const& _name) const
|
||||||
@ -129,6 +142,12 @@ vector<ASTString> DeclarationContainer::similarNames(ASTString const& _name) con
|
|||||||
if (stringWithinDistance(_name, declarationName, MAXIMUM_EDIT_DISTANCE))
|
if (stringWithinDistance(_name, declarationName, MAXIMUM_EDIT_DISTANCE))
|
||||||
similar.push_back(declarationName);
|
similar.push_back(declarationName);
|
||||||
}
|
}
|
||||||
|
for (auto const& declaration: m_invisibleDeclarations)
|
||||||
|
{
|
||||||
|
string const& declarationName = declaration.first;
|
||||||
|
if (stringWithinDistance(_name, declarationName, MAXIMUM_EDIT_DISTANCE))
|
||||||
|
similar.push_back(declarationName);
|
||||||
|
}
|
||||||
|
|
||||||
if (m_enclosingContainer)
|
if (m_enclosingContainer)
|
||||||
similar += m_enclosingContainer->similarNames(_name);
|
similar += m_enclosingContainer->similarNames(_name);
|
||||||
|
@ -51,13 +51,17 @@ public:
|
|||||||
/// @param _update if true, replaces a potential declaration that is already present
|
/// @param _update if true, replaces a potential declaration that is already present
|
||||||
/// @returns false if the name was already declared.
|
/// @returns false if the name was already declared.
|
||||||
bool registerDeclaration(Declaration const& _declaration, ASTString const* _name = nullptr, bool _invisible = false, bool _update = false);
|
bool registerDeclaration(Declaration const& _declaration, ASTString const* _name = nullptr, bool _invisible = false, bool _update = false);
|
||||||
std::vector<Declaration const*> resolveName(ASTString const& _name, bool _recursive = false) const;
|
std::vector<Declaration const*> resolveName(ASTString const& _name, bool _recursive = false, bool _alsoInvisible = false) const;
|
||||||
ASTNode const* enclosingNode() const { return m_enclosingNode; }
|
ASTNode const* enclosingNode() const { return m_enclosingNode; }
|
||||||
DeclarationContainer const* enclosingContainer() const { return m_enclosingContainer; }
|
DeclarationContainer const* enclosingContainer() const { return m_enclosingContainer; }
|
||||||
std::map<ASTString, std::vector<Declaration const*>> const& declarations() const { return m_declarations; }
|
std::map<ASTString, std::vector<Declaration const*>> const& declarations() const { return m_declarations; }
|
||||||
/// @returns whether declaration is valid, and if not also returns previous declaration.
|
/// @returns whether declaration is valid, and if not also returns previous declaration.
|
||||||
Declaration const* conflictingDeclaration(Declaration const& _declaration, ASTString const* _name = nullptr) const;
|
Declaration const* conflictingDeclaration(Declaration const& _declaration, ASTString const* _name = nullptr) const;
|
||||||
|
|
||||||
|
/// Activates a previously inactive (invisible) variable. To be used in C99 scpoing for
|
||||||
|
/// VariableDeclarationStatements.
|
||||||
|
void activateVariable(ASTString const& _name);
|
||||||
|
|
||||||
/// @returns existing declaration names similar to @a _name.
|
/// @returns existing declaration names similar to @a _name.
|
||||||
/// Searches this and all parent containers.
|
/// Searches this and all parent containers.
|
||||||
std::vector<ASTString> similarNames(ASTString const& _name) const;
|
std::vector<ASTString> similarNames(ASTString const& _name) const;
|
||||||
|
@ -50,12 +50,13 @@ NameAndTypeResolver::NameAndTypeResolver(
|
|||||||
m_scopes[nullptr]->registerDeclaration(*declaration);
|
m_scopes[nullptr]->registerDeclaration(*declaration);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool NameAndTypeResolver::registerDeclarations(ASTNode& _sourceUnit, ASTNode const* _currentScope)
|
bool NameAndTypeResolver::registerDeclarations(SourceUnit& _sourceUnit, ASTNode const* _currentScope)
|
||||||
{
|
{
|
||||||
|
bool useC99Scoping = _sourceUnit.annotation().experimentalFeatures.count(ExperimentalFeature::V050);
|
||||||
// The helper registers all declarations in m_scopes as a side-effect of its construction.
|
// The helper registers all declarations in m_scopes as a side-effect of its construction.
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
DeclarationRegistrationHelper registrar(m_scopes, _sourceUnit, m_errorReporter, _currentScope);
|
DeclarationRegistrationHelper registrar(m_scopes, _sourceUnit, useC99Scoping, m_errorReporter, _currentScope);
|
||||||
}
|
}
|
||||||
catch (FatalError const&)
|
catch (FatalError const&)
|
||||||
{
|
{
|
||||||
@ -106,7 +107,7 @@ bool NameAndTypeResolver::performImports(SourceUnit& _sourceUnit, map<string, So
|
|||||||
else
|
else
|
||||||
for (Declaration const* declaration: declarations)
|
for (Declaration const* declaration: declarations)
|
||||||
if (!DeclarationRegistrationHelper::registerDeclaration(
|
if (!DeclarationRegistrationHelper::registerDeclaration(
|
||||||
target, *declaration, alias.second.get(), &imp->location(), true, m_errorReporter
|
target, *declaration, alias.second.get(), &imp->location(), true, false, m_errorReporter
|
||||||
))
|
))
|
||||||
error = true;
|
error = true;
|
||||||
}
|
}
|
||||||
@ -114,7 +115,7 @@ bool NameAndTypeResolver::performImports(SourceUnit& _sourceUnit, map<string, So
|
|||||||
for (auto const& nameAndDeclaration: scope->second->declarations())
|
for (auto const& nameAndDeclaration: scope->second->declarations())
|
||||||
for (auto const& declaration: nameAndDeclaration.second)
|
for (auto const& declaration: nameAndDeclaration.second)
|
||||||
if (!DeclarationRegistrationHelper::registerDeclaration(
|
if (!DeclarationRegistrationHelper::registerDeclaration(
|
||||||
target, *declaration, &nameAndDeclaration.first, &imp->location(), true, m_errorReporter
|
target, *declaration, &nameAndDeclaration.first, &imp->location(), true, false, m_errorReporter
|
||||||
))
|
))
|
||||||
error = true;
|
error = true;
|
||||||
}
|
}
|
||||||
@ -151,6 +152,12 @@ bool NameAndTypeResolver::updateDeclaration(Declaration const& _declaration)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void NameAndTypeResolver::activateVariable(string const& _name)
|
||||||
|
{
|
||||||
|
solAssert(m_currentScope, "");
|
||||||
|
m_currentScope->activateVariable(_name);
|
||||||
|
}
|
||||||
|
|
||||||
vector<Declaration const*> NameAndTypeResolver::resolveName(ASTString const& _name, ASTNode const* _scope) const
|
vector<Declaration const*> NameAndTypeResolver::resolveName(ASTString const& _name, ASTNode const* _scope) const
|
||||||
{
|
{
|
||||||
auto iterator = m_scopes.find(_scope);
|
auto iterator = m_scopes.find(_scope);
|
||||||
@ -159,9 +166,9 @@ vector<Declaration const*> NameAndTypeResolver::resolveName(ASTString const& _na
|
|||||||
return iterator->second->resolveName(_name, false);
|
return iterator->second->resolveName(_name, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<Declaration const*> NameAndTypeResolver::nameFromCurrentScope(ASTString const& _name) const
|
vector<Declaration const*> NameAndTypeResolver::nameFromCurrentScope(ASTString const& _name, bool _includeInvisibles) const
|
||||||
{
|
{
|
||||||
return m_currentScope->resolveName(_name, true);
|
return m_currentScope->resolveName(_name, true, _includeInvisibles);
|
||||||
}
|
}
|
||||||
|
|
||||||
Declaration const* NameAndTypeResolver::pathFromCurrentScope(vector<ASTString> const& _path) const
|
Declaration const* NameAndTypeResolver::pathFromCurrentScope(vector<ASTString> const& _path) const
|
||||||
@ -229,7 +236,7 @@ void NameAndTypeResolver::warnVariablesNamedLikeInstructions()
|
|||||||
for (auto const& instruction: c_instructions)
|
for (auto const& instruction: c_instructions)
|
||||||
{
|
{
|
||||||
string const instructionName{boost::algorithm::to_lower_copy(instruction.first)};
|
string const instructionName{boost::algorithm::to_lower_copy(instruction.first)};
|
||||||
auto declarations = nameFromCurrentScope(instructionName);
|
auto declarations = nameFromCurrentScope(instructionName, true);
|
||||||
for (Declaration const* const declaration: declarations)
|
for (Declaration const* const declaration: declarations)
|
||||||
{
|
{
|
||||||
solAssert(!!declaration, "");
|
solAssert(!!declaration, "");
|
||||||
@ -439,9 +446,11 @@ string NameAndTypeResolver::similarNameSuggestions(ASTString const& _name) const
|
|||||||
DeclarationRegistrationHelper::DeclarationRegistrationHelper(
|
DeclarationRegistrationHelper::DeclarationRegistrationHelper(
|
||||||
map<ASTNode const*, shared_ptr<DeclarationContainer>>& _scopes,
|
map<ASTNode const*, shared_ptr<DeclarationContainer>>& _scopes,
|
||||||
ASTNode& _astRoot,
|
ASTNode& _astRoot,
|
||||||
|
bool _useC99Scoping,
|
||||||
ErrorReporter& _errorReporter,
|
ErrorReporter& _errorReporter,
|
||||||
ASTNode const* _currentScope
|
ASTNode const* _currentScope
|
||||||
):
|
):
|
||||||
|
m_useC99Scoping(_useC99Scoping),
|
||||||
m_scopes(_scopes),
|
m_scopes(_scopes),
|
||||||
m_currentScope(_currentScope),
|
m_currentScope(_currentScope),
|
||||||
m_errorReporter(_errorReporter)
|
m_errorReporter(_errorReporter)
|
||||||
@ -456,6 +465,7 @@ bool DeclarationRegistrationHelper::registerDeclaration(
|
|||||||
string const* _name,
|
string const* _name,
|
||||||
SourceLocation const* _errorLocation,
|
SourceLocation const* _errorLocation,
|
||||||
bool _warnOnShadow,
|
bool _warnOnShadow,
|
||||||
|
bool _inactive,
|
||||||
ErrorReporter& _errorReporter
|
ErrorReporter& _errorReporter
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
@ -465,10 +475,13 @@ bool DeclarationRegistrationHelper::registerDeclaration(
|
|||||||
string name = _name ? *_name : _declaration.name();
|
string name = _name ? *_name : _declaration.name();
|
||||||
Declaration const* shadowedDeclaration = nullptr;
|
Declaration const* shadowedDeclaration = nullptr;
|
||||||
if (_warnOnShadow && !name.empty() && _container.enclosingContainer())
|
if (_warnOnShadow && !name.empty() && _container.enclosingContainer())
|
||||||
for (auto const* decl: _container.enclosingContainer()->resolveName(name, true))
|
for (auto const* decl: _container.enclosingContainer()->resolveName(name, true, true))
|
||||||
shadowedDeclaration = decl;
|
shadowedDeclaration = decl;
|
||||||
|
|
||||||
if (!_container.registerDeclaration(_declaration, _name, !_declaration.isVisibleInContract()))
|
// We use "invisible" for both inactive variables in blocks and for members invisible in contracts.
|
||||||
|
// They cannot both be true at the same time.
|
||||||
|
solAssert(!(_inactive && !_declaration.isVisibleInContract()), "");
|
||||||
|
if (!_container.registerDeclaration(_declaration, _name, !_declaration.isVisibleInContract() || _inactive))
|
||||||
{
|
{
|
||||||
SourceLocation firstDeclarationLocation;
|
SourceLocation firstDeclarationLocation;
|
||||||
SourceLocation secondDeclarationLocation;
|
SourceLocation secondDeclarationLocation;
|
||||||
@ -613,32 +626,28 @@ void DeclarationRegistrationHelper::endVisit(ModifierDefinition&)
|
|||||||
bool DeclarationRegistrationHelper::visit(Block& _block)
|
bool DeclarationRegistrationHelper::visit(Block& _block)
|
||||||
{
|
{
|
||||||
_block.setScope(m_currentScope);
|
_block.setScope(m_currentScope);
|
||||||
// Enable C99-scoped variables.
|
if (m_useC99Scoping)
|
||||||
if (_block.sourceUnit().annotation().experimentalFeatures.count(ExperimentalFeature::V050))
|
|
||||||
enterNewSubScope(_block);
|
enterNewSubScope(_block);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeclarationRegistrationHelper::endVisit(Block& _block)
|
void DeclarationRegistrationHelper::endVisit(Block&)
|
||||||
{
|
{
|
||||||
// Enable C99-scoped variables.
|
if (m_useC99Scoping)
|
||||||
if (_block.sourceUnit().annotation().experimentalFeatures.count(ExperimentalFeature::V050))
|
|
||||||
closeCurrentScope();
|
closeCurrentScope();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DeclarationRegistrationHelper::visit(ForStatement& _for)
|
bool DeclarationRegistrationHelper::visit(ForStatement& _for)
|
||||||
{
|
{
|
||||||
_for.setScope(m_currentScope);
|
_for.setScope(m_currentScope);
|
||||||
if (_for.sourceUnit().annotation().experimentalFeatures.count(ExperimentalFeature::V050))
|
if (m_useC99Scoping)
|
||||||
// TODO special scoping rules for the init statement - if it is a block, then it should
|
|
||||||
// not open its own scope.
|
|
||||||
enterNewSubScope(_for);
|
enterNewSubScope(_for);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeclarationRegistrationHelper::endVisit(ForStatement& _for)
|
void DeclarationRegistrationHelper::endVisit(ForStatement&)
|
||||||
{
|
{
|
||||||
if (_for.sourceUnit().annotation().experimentalFeatures.count(ExperimentalFeature::V050))
|
if (m_useC99Scoping)
|
||||||
closeCurrentScope();
|
closeCurrentScope();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -704,7 +713,12 @@ void DeclarationRegistrationHelper::registerDeclaration(Declaration& _declaratio
|
|||||||
if (fun->isConstructor())
|
if (fun->isConstructor())
|
||||||
warnAboutShadowing = false;
|
warnAboutShadowing = false;
|
||||||
|
|
||||||
registerDeclaration(*m_scopes[m_currentScope], _declaration, nullptr, nullptr, warnAboutShadowing, m_errorReporter);
|
// Register declaration as inactive if we are in block scope and C99 mode.
|
||||||
|
bool inactive =
|
||||||
|
m_useC99Scoping &&
|
||||||
|
(dynamic_cast<Block const*>(m_currentScope) || dynamic_cast<ForStatement const*>(m_currentScope));
|
||||||
|
|
||||||
|
registerDeclaration(*m_scopes[m_currentScope], _declaration, nullptr, nullptr, warnAboutShadowing, inactive, m_errorReporter);
|
||||||
|
|
||||||
_declaration.setScope(m_currentScope);
|
_declaration.setScope(m_currentScope);
|
||||||
if (_opensScope)
|
if (_opensScope)
|
||||||
|
@ -56,7 +56,7 @@ public:
|
|||||||
/// @returns false in case of error.
|
/// @returns false in case of error.
|
||||||
/// @param _currentScope should be nullptr but can be used to inject new declarations into
|
/// @param _currentScope should be nullptr but can be used to inject new declarations into
|
||||||
/// existing scopes, used by the snippets feature.
|
/// existing scopes, used by the snippets feature.
|
||||||
bool registerDeclarations(ASTNode& _sourceUnit, ASTNode const* _currentScope = nullptr);
|
bool registerDeclarations(SourceUnit& _sourceUnit, ASTNode const* _currentScope = nullptr);
|
||||||
/// Applies the effect of import directives.
|
/// Applies the effect of import directives.
|
||||||
bool performImports(SourceUnit& _sourceUnit, std::map<std::string, SourceUnit const*> const& _sourceUnits);
|
bool performImports(SourceUnit& _sourceUnit, std::map<std::string, SourceUnit const*> const& _sourceUnits);
|
||||||
/// Resolves all names and types referenced from the given AST Node.
|
/// Resolves all names and types referenced from the given AST Node.
|
||||||
@ -69,6 +69,9 @@ public:
|
|||||||
/// that create their own scope.
|
/// that create their own scope.
|
||||||
/// @returns false in case of error.
|
/// @returns false in case of error.
|
||||||
bool updateDeclaration(Declaration const& _declaration);
|
bool updateDeclaration(Declaration const& _declaration);
|
||||||
|
/// Activates a previously inactive (invisible) variable. To be used in C99 scpoing for
|
||||||
|
/// VariableDeclarationStatements.
|
||||||
|
void activateVariable(std::string const& _name);
|
||||||
|
|
||||||
/// Resolves the given @a _name inside the scope @a _scope. If @a _scope is omitted,
|
/// Resolves the given @a _name inside the scope @a _scope. If @a _scope is omitted,
|
||||||
/// the global scope is used (i.e. the one containing only the pre-defined global variables).
|
/// the global scope is used (i.e. the one containing only the pre-defined global variables).
|
||||||
@ -78,7 +81,7 @@ public:
|
|||||||
|
|
||||||
/// Resolves a name in the "current" scope, but also searches parent scopes.
|
/// Resolves a name in the "current" scope, but also searches parent scopes.
|
||||||
/// Should only be called during the initial resolving phase.
|
/// Should only be called during the initial resolving phase.
|
||||||
std::vector<Declaration const*> nameFromCurrentScope(ASTString const& _name) const;
|
std::vector<Declaration const*> nameFromCurrentScope(ASTString const& _name, bool _includeInvisibles = false) const;
|
||||||
|
|
||||||
/// Resolves a path starting from the "current" scope, but also searches parent scopes.
|
/// Resolves a path starting from the "current" scope, but also searches parent scopes.
|
||||||
/// Should only be called during the initial resolving phase.
|
/// Should only be called during the initial resolving phase.
|
||||||
@ -139,6 +142,7 @@ public:
|
|||||||
DeclarationRegistrationHelper(
|
DeclarationRegistrationHelper(
|
||||||
std::map<ASTNode const*, std::shared_ptr<DeclarationContainer>>& _scopes,
|
std::map<ASTNode const*, std::shared_ptr<DeclarationContainer>>& _scopes,
|
||||||
ASTNode& _astRoot,
|
ASTNode& _astRoot,
|
||||||
|
bool _useC99Scoping,
|
||||||
ErrorReporter& _errorReporter,
|
ErrorReporter& _errorReporter,
|
||||||
ASTNode const* _currentScope = nullptr
|
ASTNode const* _currentScope = nullptr
|
||||||
);
|
);
|
||||||
@ -149,6 +153,7 @@ public:
|
|||||||
std::string const* _name,
|
std::string const* _name,
|
||||||
SourceLocation const* _errorLocation,
|
SourceLocation const* _errorLocation,
|
||||||
bool _warnOnShadow,
|
bool _warnOnShadow,
|
||||||
|
bool _inactive,
|
||||||
ErrorReporter& _errorReporter
|
ErrorReporter& _errorReporter
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -185,6 +190,7 @@ private:
|
|||||||
/// @returns the canonical name of the current scope.
|
/// @returns the canonical name of the current scope.
|
||||||
std::string currentCanonicalName() const;
|
std::string currentCanonicalName() const;
|
||||||
|
|
||||||
|
bool m_useC99Scoping = false;
|
||||||
std::map<ASTNode const*, std::shared_ptr<DeclarationContainer>>& m_scopes;
|
std::map<ASTNode const*, std::shared_ptr<DeclarationContainer>>& m_scopes;
|
||||||
ASTNode const* m_currentScope = nullptr;
|
ASTNode const* m_currentScope = nullptr;
|
||||||
VariableScope* m_currentFunction = nullptr;
|
VariableScope* m_currentFunction = nullptr;
|
||||||
|
@ -83,6 +83,16 @@ void ReferencesResolver::endVisit(ForStatement const& _for)
|
|||||||
m_resolver.setScope(_for.scope());
|
m_resolver.setScope(_for.scope());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ReferencesResolver::endVisit(VariableDeclarationStatement const& _varDeclStatement)
|
||||||
|
{
|
||||||
|
if (!m_resolveInsideCode)
|
||||||
|
return;
|
||||||
|
if (m_experimental050Mode)
|
||||||
|
for (auto const& var: _varDeclStatement.declarations())
|
||||||
|
if (var)
|
||||||
|
m_resolver.activateVariable(var->name());
|
||||||
|
}
|
||||||
|
|
||||||
bool ReferencesResolver::visit(Identifier const& _identifier)
|
bool ReferencesResolver::visit(Identifier const& _identifier)
|
||||||
{
|
{
|
||||||
auto declarations = m_resolver.nameFromCurrentScope(_identifier.name());
|
auto declarations = m_resolver.nameFromCurrentScope(_identifier.name());
|
||||||
|
@ -61,6 +61,7 @@ private:
|
|||||||
virtual void endVisit(Block const& _block) override;
|
virtual void endVisit(Block const& _block) override;
|
||||||
virtual bool visit(ForStatement const& _for) override;
|
virtual bool visit(ForStatement const& _for) override;
|
||||||
virtual void endVisit(ForStatement const& _for) override;
|
virtual void endVisit(ForStatement const& _for) override;
|
||||||
|
virtual void endVisit(VariableDeclarationStatement const& _varDeclStatement) override;
|
||||||
virtual bool visit(Identifier const& _identifier) override;
|
virtual bool visit(Identifier const& _identifier) override;
|
||||||
virtual bool visit(ElementaryTypeName const& _typeName) override;
|
virtual bool visit(ElementaryTypeName const& _typeName) override;
|
||||||
virtual bool visit(FunctionDefinition const& _functionDefinition) override;
|
virtual bool visit(FunctionDefinition const& _functionDefinition) override;
|
||||||
|
@ -284,6 +284,26 @@ BOOST_AUTO_TEST_CASE(conditional_expression_functions)
|
|||||||
ABI_CHECK(callContractFunction("f(bool)", false), encodeArgs(u256(2)));
|
ABI_CHECK(callContractFunction("f(bool)", false), encodeArgs(u256(2)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(C99_scoping_activation)
|
||||||
|
{
|
||||||
|
char const* sourceCode = R"(
|
||||||
|
pragma experimental "v0.5.0";
|
||||||
|
contract test {
|
||||||
|
function f() pure public returns (uint) {
|
||||||
|
uint x = 7;
|
||||||
|
{
|
||||||
|
x = 3; // This should still assign to the outer variable
|
||||||
|
uint x;
|
||||||
|
x = 4; // This should assign to the new one
|
||||||
|
}
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
compileAndRun(sourceCode);
|
||||||
|
ABI_CHECK(callContractFunction("f()"), encodeArgs(3));
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(recursive_calls)
|
BOOST_AUTO_TEST_CASE(recursive_calls)
|
||||||
{
|
{
|
||||||
char const* sourceCode = R"(
|
char const* sourceCode = R"(
|
||||||
|
@ -135,19 +135,59 @@ BOOST_AUTO_TEST_CASE(scoping)
|
|||||||
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(scoping_for)
|
BOOST_AUTO_TEST_CASE(scoping_activation_old)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
contract test {
|
||||||
|
function f() pure public {
|
||||||
|
x = 3;
|
||||||
|
uint x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_SUCCESS_NO_WARNINGS(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(scoping_activation)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
pragma experimental "v0.5.0";
|
||||||
|
contract test {
|
||||||
|
function f() pure public {
|
||||||
|
x = 3;
|
||||||
|
uint x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(scoping_self_use)
|
||||||
{
|
{
|
||||||
char const* text = R"(
|
char const* text = R"(
|
||||||
pragma experimental "v0.5.0";
|
pragma experimental "v0.5.0";
|
||||||
contract test {
|
contract test {
|
||||||
function f() public {
|
function f() public {
|
||||||
|
uint a = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(scoping_for)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
pragma experimental "v0.5.0";
|
||||||
|
contract test {
|
||||||
|
function f() pure public {
|
||||||
for (uint x = 0; x < 10; x ++){
|
for (uint x = 0; x < 10; x ++){
|
||||||
x = 2;
|
x = 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
CHECK_SUCCESS(text);
|
CHECK_WARNING(text, "Experimental features");
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(scoping_for2)
|
BOOST_AUTO_TEST_CASE(scoping_for2)
|
||||||
@ -155,7 +195,21 @@ BOOST_AUTO_TEST_CASE(scoping_for2)
|
|||||||
char const* text = R"(
|
char const* text = R"(
|
||||||
pragma experimental "v0.5.0";
|
pragma experimental "v0.5.0";
|
||||||
contract test {
|
contract test {
|
||||||
function f() public {
|
function f() pure public {
|
||||||
|
for (uint x = 0; x < 10; x ++)
|
||||||
|
x = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_WARNING(text, "Experimental features");
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(scoping_for3)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
pragma experimental "v0.5.0";
|
||||||
|
contract test {
|
||||||
|
function f() pure public {
|
||||||
for (uint x = 0; x < 10; x ++){
|
for (uint x = 0; x < 10; x ++){
|
||||||
x = 2;
|
x = 2;
|
||||||
}
|
}
|
||||||
@ -166,6 +220,21 @@ BOOST_AUTO_TEST_CASE(scoping_for2)
|
|||||||
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(scoping_for_decl_in_body)
|
||||||
|
{
|
||||||
|
char const* text = R"(
|
||||||
|
pragma experimental "v0.5.0";
|
||||||
|
contract test {
|
||||||
|
function f() pure public {
|
||||||
|
for (;; y++){
|
||||||
|
uint y = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_ERROR(text, DeclarationError, "Undeclared identifier");
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(name_shadowing)
|
BOOST_AUTO_TEST_CASE(name_shadowing)
|
||||||
{
|
{
|
||||||
char const* text = R"(
|
char const* text = R"(
|
||||||
|
Loading…
Reference in New Issue
Block a user