forked from OSchip/llvm-project
Implement support for the __is_final type trait, to determine whether
a class is marked 'final', from Alberto Ganesh Barbati! Fixes PR11462. llvm-svn: 145775
This commit is contained in:
parent
dd5eb9df0c
commit
dca70af22f
|
@ -724,6 +724,7 @@ struct is_convertible_to {
|
|||
<li><code>__is_polymorphic</code> (GNU, Microsoft)</li>
|
||||
<li><code>__is_union</code> (GNU, Microsoft)</li>
|
||||
<li><code>__is_literal(type)</code>: Determines whether the given type is a literal type</li>
|
||||
<li><code>__is_final</code>: Determines whether the given type is declared with a <code>final</code> class-virt-specifier.</li>
|
||||
<li><code>__underlying_type(type)</code>: Retrieves the underlying type for a given <code>enum</code> type. This trait is required to implement the C++11 standard library.</li>
|
||||
</ul>
|
||||
|
||||
|
|
|
@ -353,6 +353,7 @@ KEYWORD(__is_class , KEYCXX)
|
|||
KEYWORD(__is_convertible_to , KEYCXX)
|
||||
KEYWORD(__is_empty , KEYCXX)
|
||||
KEYWORD(__is_enum , KEYCXX)
|
||||
KEYWORD(__is_final , KEYCXX)
|
||||
// Tentative name - there's no implementation of std::is_literal_type yet.
|
||||
KEYWORD(__is_literal , KEYCXX)
|
||||
// Name for GCC 4.6 compatibility - people have already written libraries using
|
||||
|
|
|
@ -35,6 +35,7 @@ namespace clang {
|
|||
UTT_IsConst,
|
||||
UTT_IsEmpty,
|
||||
UTT_IsEnum,
|
||||
UTT_IsFinal,
|
||||
UTT_IsFloatingPoint,
|
||||
UTT_IsFunction,
|
||||
UTT_IsFundamental,
|
||||
|
|
|
@ -1409,6 +1409,7 @@ static const char *getTypeTraitName(UnaryTypeTrait UTT) {
|
|||
case UTT_IsConst: return "__is_const";
|
||||
case UTT_IsEmpty: return "__is_empty";
|
||||
case UTT_IsEnum: return "__is_enum";
|
||||
case UTT_IsFinal: return "__is_final";
|
||||
case UTT_IsFloatingPoint: return "__is_floating_point";
|
||||
case UTT_IsFunction: return "__is_function";
|
||||
case UTT_IsFundamental: return "__is_fundamental";
|
||||
|
|
|
@ -675,6 +675,7 @@ static bool HasFeature(const Preprocessor &PP, const IdentifierInfo *II) {
|
|||
PP.getIdentifierInfo("__is_empty")->getTokenID()
|
||||
!= tok::identifier)
|
||||
.Case("is_enum", LangOpts.CPlusPlus)
|
||||
.Case("is_final", LangOpts.CPlusPlus)
|
||||
.Case("is_literal", LangOpts.CPlusPlus)
|
||||
.Case("is_standard_layout", LangOpts.CPlusPlus)
|
||||
// __is_pod is available only if the horrible
|
||||
|
|
|
@ -565,6 +565,7 @@ ExprResult Parser::ParseCastExpression(bool isUnaryExpression,
|
|||
/// '__is_class'
|
||||
/// '__is_empty' [TODO]
|
||||
/// '__is_enum'
|
||||
/// '__is_final'
|
||||
/// '__is_pod'
|
||||
/// '__is_polymorphic'
|
||||
/// '__is_trivial'
|
||||
|
@ -1087,6 +1088,7 @@ ExprResult Parser::ParseCastExpression(bool isUnaryExpression,
|
|||
case tok::kw___is_trivial:
|
||||
case tok::kw___is_trivially_copyable:
|
||||
case tok::kw___is_union:
|
||||
case tok::kw___is_final:
|
||||
case tok::kw___has_trivial_constructor:
|
||||
case tok::kw___has_trivial_copy:
|
||||
case tok::kw___has_trivial_assign:
|
||||
|
|
|
@ -2241,6 +2241,7 @@ static UnaryTypeTrait UnaryTypeTraitFromTokKind(tok::TokenKind kind) {
|
|||
case tok::kw___is_const: return UTT_IsConst;
|
||||
case tok::kw___is_empty: return UTT_IsEmpty;
|
||||
case tok::kw___is_enum: return UTT_IsEnum;
|
||||
case tok::kw___is_final: return UTT_IsFinal;
|
||||
case tok::kw___is_floating_point: return UTT_IsFloatingPoint;
|
||||
case tok::kw___is_function: return UTT_IsFunction;
|
||||
case tok::kw___is_fundamental: return UTT_IsFundamental;
|
||||
|
|
|
@ -670,6 +670,7 @@ Parser::isExpressionOrTypeSpecifierSimple(tok::TokenKind Kind) {
|
|||
case tok::kw___is_convertible_to:
|
||||
case tok::kw___is_empty:
|
||||
case tok::kw___is_enum:
|
||||
case tok::kw___is_final:
|
||||
case tok::kw___is_literal:
|
||||
case tok::kw___is_literal_type:
|
||||
case tok::kw___is_pod:
|
||||
|
|
|
@ -2650,6 +2650,9 @@ static bool CheckUnaryTypeTraitTypeCompleteness(Sema &S,
|
|||
case UTT_IsAbstract:
|
||||
// Fall-through
|
||||
|
||||
// These traits require a complete type.
|
||||
case UTT_IsFinal:
|
||||
|
||||
// These trait expressions are designed to help implement predicates in
|
||||
// [meta.unary.prop] despite not being named the same. They are specified
|
||||
// by both GCC and the Embarcadero C++ compiler, and require the complete
|
||||
|
@ -2775,6 +2778,10 @@ static bool EvaluateUnaryTypeTrait(Sema &Self, UnaryTypeTrait UTT,
|
|||
if (const CXXRecordDecl *RD = T->getAsCXXRecordDecl())
|
||||
return RD->isAbstract();
|
||||
return false;
|
||||
case UTT_IsFinal:
|
||||
if (const CXXRecordDecl *RD = T->getAsCXXRecordDecl())
|
||||
return RD->hasAttr<FinalAttr>();
|
||||
return false;
|
||||
case UTT_IsSigned:
|
||||
return T->isSignedIntegerType();
|
||||
case UTT_IsUnsigned:
|
||||
|
|
|
@ -70,6 +70,11 @@ int is_enum();
|
|||
#endif
|
||||
// CHECK: int is_enum();
|
||||
|
||||
#if __has_feature(is_final)
|
||||
int is_final();
|
||||
#endif
|
||||
// CHECK: int is_final();
|
||||
|
||||
#if __has_feature(is_pod)
|
||||
int is_pod();
|
||||
#endif
|
||||
|
|
|
@ -235,6 +235,37 @@ void is_enum()
|
|||
{ int arr[F(__is_enum(HasAnonymousUnion))]; }
|
||||
}
|
||||
|
||||
struct FinalClass final {
|
||||
};
|
||||
|
||||
template<typename T>
|
||||
struct PotentiallyFinal { };
|
||||
|
||||
template<typename T>
|
||||
struct PotentiallyFinal<T*> final { };
|
||||
|
||||
template<>
|
||||
struct PotentiallyFinal<int> final { };
|
||||
|
||||
void is_final()
|
||||
{
|
||||
{ int arr[T(__is_final(FinalClass))]; }
|
||||
{ int arr[T(__is_final(PotentiallyFinal<float*>))]; }
|
||||
{ int arr[T(__is_final(PotentiallyFinal<int>))]; }
|
||||
|
||||
{ int arr[F(__is_final(int))]; }
|
||||
{ int arr[F(__is_final(Union))]; }
|
||||
{ int arr[F(__is_final(Int))]; }
|
||||
{ int arr[F(__is_final(IntAr))]; }
|
||||
{ int arr[F(__is_final(UnionAr))]; }
|
||||
{ int arr[F(__is_final(Derives))]; }
|
||||
{ int arr[F(__is_final(ClassType))]; }
|
||||
{ int arr[F(__is_final(cvoid))]; }
|
||||
{ int arr[F(__is_final(IntArNB))]; }
|
||||
{ int arr[F(__is_final(HasAnonymousUnion))]; }
|
||||
{ int arr[F(__is_final(PotentiallyFinal<float>))]; }
|
||||
}
|
||||
|
||||
typedef HasVirt Polymorph;
|
||||
struct InheritPolymorph : Polymorph {};
|
||||
|
||||
|
|
Loading…
Reference in New Issue