mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-18 19:16:43 +00:00
[clang] Make nullability-on-classes more robust to redeclarations (#114778)
This is relevant after b24650e814e55d90acfc40acf045456c98f32b9c where the selected template decl can be anything, even apparently a friend declaration in some cases.
This commit is contained in:
parent
b4ef43fc75
commit
17d8ed717f
@ -43,6 +43,7 @@
|
||||
#include "llvm/ADT/APSInt.h"
|
||||
#include "llvm/ADT/ArrayRef.h"
|
||||
#include "llvm/ADT/FoldingSet.h"
|
||||
#include "llvm/ADT/STLExtras.h"
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/Support/Casting.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
@ -4774,7 +4775,10 @@ bool Type::canHaveNullability(bool ResultIfUnknown) const {
|
||||
->getTemplateName()
|
||||
.getAsTemplateDecl())
|
||||
if (auto *CTD = dyn_cast<ClassTemplateDecl>(templateDecl))
|
||||
return CTD->getTemplatedDecl()->hasAttr<TypeNullableAttr>();
|
||||
return llvm::any_of(
|
||||
CTD->redecls(), [](const RedeclarableTemplateDecl *RTD) {
|
||||
return RTD->getTemplatedDecl()->hasAttr<TypeNullableAttr>();
|
||||
});
|
||||
return ResultIfUnknown;
|
||||
|
||||
case Type::Builtin:
|
||||
@ -4841,10 +4845,14 @@ bool Type::canHaveNullability(bool ResultIfUnknown) const {
|
||||
// For template specializations, look only at primary template attributes.
|
||||
// This is a consistent regardless of whether the instantiation is known.
|
||||
if (const auto *CTSD = dyn_cast<ClassTemplateSpecializationDecl>(RD))
|
||||
return CTSD->getSpecializedTemplate()
|
||||
->getTemplatedDecl()
|
||||
->hasAttr<TypeNullableAttr>();
|
||||
return RD->hasAttr<TypeNullableAttr>();
|
||||
return llvm::any_of(
|
||||
CTSD->getSpecializedTemplate()->redecls(),
|
||||
[](const RedeclarableTemplateDecl *RTD) {
|
||||
return RTD->getTemplatedDecl()->hasAttr<TypeNullableAttr>();
|
||||
});
|
||||
return llvm::any_of(RD->redecls(), [](const TagDecl *RD) {
|
||||
return RD->hasAttr<TypeNullableAttr>();
|
||||
});
|
||||
}
|
||||
|
||||
// Non-pointer types.
|
||||
|
27
clang/test/SemaCXX/nullability_redecl.cpp
Normal file
27
clang/test/SemaCXX/nullability_redecl.cpp
Normal file
@ -0,0 +1,27 @@
|
||||
// RUN: %clang_cc1 -std=c++11 -fsyntax-only -Wno-nullability-declspec %s -verify -Wnullable-to-nonnull-conversion -I%S/Inputs
|
||||
|
||||
class Foo;
|
||||
using Foo1 = Foo _Nonnull; // expected-error{{nullability specifier '_Nonnull' cannot be applied to non-pointer type 'Foo'}}
|
||||
class _Nullable Foo;
|
||||
using Foo2 = Foo _Nonnull;
|
||||
class Foo;
|
||||
using Foo3 = Foo _Nonnull;
|
||||
|
||||
template <class T>
|
||||
class Bar;
|
||||
using Bar1 = Bar<int> _Nonnull; // expected-error{{nullability specifier '_Nonnull' cannot be applied to non-pointer type 'Bar<int>'}}
|
||||
template <class T>
|
||||
class _Nullable Bar;
|
||||
using Bar2 = Bar<int> _Nonnull;
|
||||
template <class T>
|
||||
class Bar;
|
||||
using Bar3 = Bar<int> _Nonnull;
|
||||
|
||||
namespace std {
|
||||
template<class T> class unique_ptr;
|
||||
using UP1 = unique_ptr<int> _Nonnull;
|
||||
class X { template<class T> friend class unique_ptr; };
|
||||
using UP2 = unique_ptr<int> _Nonnull;
|
||||
template<class T> class unique_ptr;
|
||||
using UP3 = unique_ptr<int> _Nonnull;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user