From e0ac9bfbf588e269fefc8758e715ee5804affb18 Mon Sep 17 00:00:00 2001 From: Reid Kleckner Date: Tue, 12 Nov 2013 02:22:34 +0000 Subject: [PATCH] -fms-compatibility: Use C++98 null pointer constant rules Patch by Will Wilson! git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@194441 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/AST/Expr.cpp | 10 ++++++++-- test/SemaCXX/MicrosoftCompatibility.cpp | 16 ++++++++++++++++ 2 files changed, 24 insertions(+), 2 deletions(-) diff --git a/lib/AST/Expr.cpp b/lib/AST/Expr.cpp index e467883288..8bbf2ff546 100644 --- a/lib/AST/Expr.cpp +++ b/lib/AST/Expr.cpp @@ -3051,7 +3051,8 @@ bool Expr::hasNonTrivialCall(ASTContext &Ctx) { Expr::NullPointerConstantKind Expr::isNullPointerConstant(ASTContext &Ctx, NullPointerConstantValueDependence NPC) const { - if (isValueDependent() && !Ctx.getLangOpts().CPlusPlus11) { + if (isValueDependent() && + (!Ctx.getLangOpts().CPlusPlus11 || Ctx.getLangOpts().MicrosoftMode)) { switch (NPC) { case NPC_NeverValueDependent: llvm_unreachable("Unexpected value dependent expression!"); @@ -3133,8 +3134,13 @@ Expr::isNullPointerConstant(ASTContext &Ctx, if (Ctx.getLangOpts().CPlusPlus11) { // C++11 [conv.ptr]p1: A null pointer constant is an integer literal with // value zero or a prvalue of type std::nullptr_t. + // Microsoft mode permits C++98 rules reflecting MSVC behavior. const IntegerLiteral *Lit = dyn_cast(this); - return (Lit && !Lit->getValue()) ? NPCK_ZeroLiteral : NPCK_NotNull; + if (Lit && !Lit->getValue()) + return NPCK_ZeroLiteral; + else if (!Ctx.getLangOpts().MicrosoftMode || + !isCXX98IntegralConstantExpr(Ctx)) + return NPCK_NotNull; } else { // If we have an integer constant expression, we need to *evaluate* it and // test for the value 0. diff --git a/test/SemaCXX/MicrosoftCompatibility.cpp b/test/SemaCXX/MicrosoftCompatibility.cpp index 6a48f36801..05037ac6a0 100644 --- a/test/SemaCXX/MicrosoftCompatibility.cpp +++ b/test/SemaCXX/MicrosoftCompatibility.cpp @@ -178,3 +178,19 @@ namespace PR11791 { del((void*)a); // expected-note {{in instantiation of function template specialization}} } } + +namespace IntToNullPtrConv { + struct Foo { + static const int ZERO = 0; + typedef void (Foo::*MemberFcnPtr)(); + }; + + struct Bar { + const Foo::MemberFcnPtr pB; + }; + + Bar g_bar = { (Foo::MemberFcnPtr)Foo::ZERO }; + + template int *get_n() { return N; } // expected-warning {{expression which evaluates to zero treated as a null pointer constant}} + int *g_nullptr = get_n<0>(); // expected-note {{in instantiation of function template specialization}} +} -- 2.40.0