diff --git a/clang/include/clang/Basic/DiagnosticGroups.td b/clang/include/clang/Basic/DiagnosticGroups.td --- a/clang/include/clang/Basic/DiagnosticGroups.td +++ b/clang/include/clang/Basic/DiagnosticGroups.td @@ -1207,6 +1207,7 @@ def MicrosoftInaccessibleBase : DiagGroup<"microsoft-inaccessible-base">; def MicrosoftStaticAssert : DiagGroup<"microsoft-static-assert">; def MicrosoftInitFromPredefined : DiagGroup<"microsoft-init-from-predefined">; +def MicrosoftConcatPredefined : DiagGroup<"microsoft-concat-predefined">; // Aliases. def : DiagGroup<"msvc-include", [MicrosoftInclude]>; @@ -1224,7 +1225,8 @@ MicrosoftFlexibleArray, MicrosoftExtraQualification, MicrosoftCast, MicrosoftConstInit, MicrosoftVoidPseudoDtor, MicrosoftAnonTag, MicrosoftCommentPaste, MicrosoftEndOfFile, MicrosoftStaticAssert, - MicrosoftInitFromPredefined, MicrosoftInconsistentDllImport]>; + MicrosoftInitFromPredefined, MicrosoftConcatPredefined, + MicrosoftInconsistentDllImport]>; def ClangClPch : DiagGroup<"clang-cl-pch">; diff --git a/clang/include/clang/Basic/DiagnosticSemaKinds.td b/clang/include/clang/Basic/DiagnosticSemaKinds.td --- a/clang/include/clang/Basic/DiagnosticSemaKinds.td +++ b/clang/include/clang/Basic/DiagnosticSemaKinds.td @@ -115,6 +115,9 @@ def ext_init_from_predefined : ExtWarn< "initializing an array from a '%0' predefined identifier is a Microsoft extension">, InGroup; +def ext_concat_predef_ms : ExtWarn< + "string literal concatenation of predefined identifier '%0' is a Microsoft extension">, + InGroup; def warn_float_overflow : Warning< "magnitude of floating-point constant too large for type %0; maximum is %1">, InGroup; diff --git a/clang/include/clang/Basic/TokenKinds.h b/clang/include/clang/Basic/TokenKinds.h --- a/clang/include/clang/Basic/TokenKinds.h +++ b/clang/include/clang/Basic/TokenKinds.h @@ -19,6 +19,8 @@ namespace clang { +class LangOptions; + namespace tok { /// Provides a simple uniform namespace for tokens from all C languages. @@ -107,6 +109,11 @@ /// Return true if this is an annotation token representing a pragma. bool isPragmaAnnotation(TokenKind K); +// Return true if the token corresponds to a function local predefined macro, +// which expands to a string literal, that can be concatenated with other +// string literals (only in Microsoft mode). +bool isFunctionLocalStringLiteralMacro(TokenKind K, const LangOptions &LO); + inline constexpr bool isRegularKeywordAttribute(TokenKind K) { return (false #define KEYWORD_ATTRIBUTE(X) || (K == tok::kw_##X) diff --git a/clang/include/clang/Sema/Sema.h b/clang/include/clang/Sema/Sema.h --- a/clang/include/clang/Sema/Sema.h +++ b/clang/include/clang/Sema/Sema.h @@ -3598,6 +3598,10 @@ /// in a 'block', this returns the containing context. NamedDecl *getCurFunctionOrMethodDecl() const; + /// getCurLocalScopeDecl - Return the Decl for either of: + /// block, lambda, captured statement, function, or nullptr. + Decl *getCurLocalScopeDecl(); + /// Add this decl to the scope shadowed decl chains. void PushOnScopeChains(NamedDecl *D, Scope *S, bool AddToContext = true); @@ -5693,6 +5697,11 @@ SourceLocation LitEndLoc, TemplateArgumentListInfo *ExplicitTemplateArgs = nullptr); + // ExpandFunctionLocalPredefinedMacros - Returns a new vector of Tokens, + // where Tokens representing function local predefined macros (such as + // __FUNCTION__) are replaced (expanded) with string-literal Tokens. + std::vector ExpandFunctionLocalPredefinedMacros(ArrayRef Toks); + ExprResult BuildPredefinedExpr(SourceLocation Loc, PredefinedExpr::IdentKind IK); ExprResult ActOnPredefinedExpr(SourceLocation Loc, tok::TokenKind Kind); diff --git a/clang/lib/Basic/TokenKinds.cpp b/clang/lib/Basic/TokenKinds.cpp --- a/clang/lib/Basic/TokenKinds.cpp +++ b/clang/lib/Basic/TokenKinds.cpp @@ -11,6 +11,8 @@ //===----------------------------------------------------------------------===// #include "clang/Basic/TokenKinds.h" + +#include "clang/Basic/LangOptions.h" #include "llvm/Support/ErrorHandling.h" using namespace clang; @@ -74,3 +76,11 @@ } return false; } + +bool tok::isFunctionLocalStringLiteralMacro(TokenKind K, + const LangOptions &LO) { + return LO.MicrosoftExt && + (K == tok::kw___FUNCTION__ || K == tok::kw_L__FUNCTION__ || + K == tok::kw___FUNCSIG__ || K == tok::kw_L__FUNCSIG__ || + K == tok::kw___FUNCDNAME__); +} diff --git a/clang/lib/Parse/ParseExpr.cpp b/clang/lib/Parse/ParseExpr.cpp --- a/clang/lib/Parse/ParseExpr.cpp +++ b/clang/lib/Parse/ParseExpr.cpp @@ -731,6 +731,12 @@ }; } +static inline bool TokenIsLikeStringLiteral(const Token &Tok, + const LangOptions &langOpts) { + return tok::isStringLiteral(Tok.getKind()) || + tok::isFunctionLocalStringLiteralMacro(Tok.getKind(), langOpts); +} + /// Parse a cast-expression, or, if \pisUnaryExpression is true, parse /// a unary-expression. /// @@ -1297,9 +1303,17 @@ case tok::kw_L__FUNCTION__: // primary-expression: L__FUNCTION__ [MS] case tok::kw_L__FUNCSIG__: // primary-expression: L__FUNCSIG__ [MS] case tok::kw___PRETTY_FUNCTION__: // primary-expression: __P..Y_F..N__ [GNU] - Res = Actions.ActOnPredefinedExpr(Tok.getLocation(), SavedKind); - ConsumeToken(); - break; + // Function local predefined macros are represented by PredefinedExpr except + // when Microsoft extensions are enabled and one of these macros is adjacent + // to a string literal or another one of these macros. + if (!(getLangOpts().MicrosoftExt && + TokenIsLikeStringLiteral(Tok, getLangOpts()) && + TokenIsLikeStringLiteral(NextToken(), getLangOpts()))) { + Res = Actions.ActOnPredefinedExpr(Tok.getLocation(), SavedKind); + ConsumeToken(); + break; + } + [[fallthrough]]; // treat MS function local macros as concatenable strings case tok::string_literal: // primary-expression: string-literal case tok::wide_string_literal: case tok::utf8_string_literal: @@ -3267,16 +3281,23 @@ ExprResult Parser::ParseStringLiteralExpression(bool AllowUserDefinedLiteral, bool Unevaluated) { - assert(isTokenStringLiteral() && "Not a string literal!"); + assert(TokenIsLikeStringLiteral(Tok, getLangOpts()) && + "Not a string-literal-like token!"); - // String concat. Note that keywords like __func__ and __FUNCTION__ are not - // considered to be strings for concatenation purposes. + // String concatenation. + // Note: some keywords like __FUNCTION__ are not considered to be strings + // for concatenation purposes, unless Microsoft extensions are enabled. SmallVector StringToks; do { StringToks.push_back(Tok); - ConsumeStringToken(); - } while (isTokenStringLiteral()); + ConsumeAnyToken(); + } while (TokenIsLikeStringLiteral(Tok, getLangOpts())); + + assert( + (StringToks.size() != 1 || + tok::isStringLiteral(StringToks[0].getKind())) && + "single predefined identifiers shall be handled by ActOnPredefinedExpr"); if (Unevaluated) { assert(!AllowUserDefinedLiteral && "UDL are always evaluated"); diff --git a/clang/lib/Sema/Sema.cpp b/clang/lib/Sema/Sema.cpp --- a/clang/lib/Sema/Sema.cpp +++ b/clang/lib/Sema/Sema.cpp @@ -1491,6 +1491,19 @@ return nullptr; } +Decl *Sema::getCurLocalScopeDecl() { + if (const BlockScopeInfo *BSI = getCurBlock()) + return BSI->TheDecl; + else if (const LambdaScopeInfo *LSI = getCurLambda()) + return LSI->CallOperator; + else if (const CapturedRegionScopeInfo *CSI = getCurCapturedRegion()) + return CSI->TheCapturedDecl; + else if (NamedDecl *ND = getCurFunctionOrMethodDecl()) + return ND; + else + return nullptr; +} + LangAS Sema::getDefaultCXXMethodAddrSpace() const { if (getLangOpts().OpenCL) return getASTContext().getDefaultOpenCLPointeeAddrSpace(); diff --git a/clang/lib/Sema/SemaExpr.cpp b/clang/lib/Sema/SemaExpr.cpp --- a/clang/lib/Sema/SemaExpr.cpp +++ b/clang/lib/Sema/SemaExpr.cpp @@ -1883,6 +1883,27 @@ ContainsUnexpandedParameterPack, ResultIndex); } +static PredefinedExpr::IdentKind getPredefinedExprKind(tok::TokenKind Kind) { + switch (Kind) { + default: + llvm_unreachable("unexpected TokenKind"); + case tok::kw___func__: + return PredefinedExpr::Func; // [C99 6.4.2.2] + case tok::kw___FUNCTION__: + return PredefinedExpr::Function; + case tok::kw___FUNCDNAME__: + return PredefinedExpr::FuncDName; // [MS] + case tok::kw___FUNCSIG__: + return PredefinedExpr::FuncSig; // [MS] + case tok::kw_L__FUNCTION__: + return PredefinedExpr::LFunction; // [MS] + case tok::kw_L__FUNCSIG__: + return PredefinedExpr::LFuncSig; // [MS] + case tok::kw___PRETTY_FUNCTION__: + return PredefinedExpr::PrettyFunction; // [GNU] + } +} + /// getUDSuffixLoc - Create a SourceLocation for a ud-suffix, given the /// location of the token and the offset of the ud-suffix within it. static SourceLocation getUDSuffixLoc(Sema &S, SourceLocation TokLoc, @@ -1946,6 +1967,55 @@ return Lit; } +std::vector +Sema::ExpandFunctionLocalPredefinedMacros(ArrayRef Toks) { + // MSVC treats some predefined identifiers (e.g. __FUNCTION__) as function + // local macros that expand to string literals that may be concatenated. + // These macros are expanded here (in Sema), because StringLiteralParser + // (in Lex) doesn't know the enclosing function (because it hasn't been + // parsed yet). + assert(getLangOpts().MicrosoftExt); + + // Note: Although function local macros are defined only inside functions, + // we ensure a valid `currentDecl` even outside of a function. This allows + // expansion of macros into empty string literals without additional checks. + Decl *currentDecl = getCurLocalScopeDecl(); + if (!currentDecl) + currentDecl = Context.getTranslationUnitDecl(); + + std::vector ExpandedToks; + ExpandedToks.reserve(Toks.size()); + for (const Token &Tok : Toks) { + if (!tok::isFunctionLocalStringLiteralMacro(Tok.getKind(), getLangOpts())) { + assert(tok::isStringLiteral(Tok.getKind())); + ExpandedToks.emplace_back(Tok); + continue; + } + if (isa(currentDecl)) { + Diag(Tok.getLocation(), diag::ext_predef_outside_function); + } + // Stringify predefined expression + Diag(Tok.getLocation(), diag::ext_concat_predef_ms) << Tok.getKind(); + SmallString<64> Str; + llvm::raw_svector_ostream OS(Str); + Token &Exp = ExpandedToks.emplace_back(); + Exp.startToken(); + if (Tok.getKind() == tok::kw_L__FUNCTION__ || + Tok.getKind() == tok::kw_L__FUNCSIG__) { + OS << 'L'; + Exp.setKind(tok::wide_string_literal); + } else { + Exp.setKind(tok::string_literal); + } + OS << "R\"(" + << PredefinedExpr::ComputeName(getPredefinedExprKind(Tok.getKind()), + currentDecl) + << ")\""; + PP.CreateString(OS.str(), Exp, Tok.getLocation(), Tok.getEndLoc()); + } + return ExpandedToks; +} + /// ActOnStringLiteral - The specified tokens were lexed as pasted string /// fragments (e.g. "foo" "bar" L"baz"). The result string has to handle string /// concatenation ([C99 5.1.1.2, translation phase #6]), so it may come from @@ -1956,6 +2026,10 @@ Sema::ActOnStringLiteral(ArrayRef StringToks, Scope *UDLScope) { assert(!StringToks.empty() && "Must have at least one string!"); + std::vector ExpandedToks; + if (getLangOpts().MicrosoftExt) + StringToks = ExpandedToks = ExpandFunctionLocalPredefinedMacros(StringToks); + StringLiteralParser Literal(StringToks, PP); if (Literal.hadError) return ExprError(); @@ -3625,17 +3699,7 @@ ExprResult Sema::BuildPredefinedExpr(SourceLocation Loc, PredefinedExpr::IdentKind IK) { - // Pick the current block, lambda, captured statement or function. - Decl *currentDecl = nullptr; - if (const BlockScopeInfo *BSI = getCurBlock()) - currentDecl = BSI->TheDecl; - else if (const LambdaScopeInfo *LSI = getCurLambda()) - currentDecl = LSI->CallOperator; - else if (const CapturedRegionScopeInfo *CSI = getCurCapturedRegion()) - currentDecl = CSI->TheCapturedDecl; - else - currentDecl = getCurFunctionOrMethodDecl(); - + Decl *currentDecl = getCurLocalScopeDecl(); if (!currentDecl) { Diag(Loc, diag::ext_predef_outside_function); currentDecl = Context.getTranslationUnitDecl(); @@ -3700,20 +3764,7 @@ } ExprResult Sema::ActOnPredefinedExpr(SourceLocation Loc, tok::TokenKind Kind) { - PredefinedExpr::IdentKind IK; - - switch (Kind) { - default: llvm_unreachable("Unknown simple primary expr!"); - case tok::kw___func__: IK = PredefinedExpr::Func; break; // [C99 6.4.2.2] - case tok::kw___FUNCTION__: IK = PredefinedExpr::Function; break; - case tok::kw___FUNCDNAME__: IK = PredefinedExpr::FuncDName; break; // [MS] - case tok::kw___FUNCSIG__: IK = PredefinedExpr::FuncSig; break; // [MS] - case tok::kw_L__FUNCTION__: IK = PredefinedExpr::LFunction; break; // [MS] - case tok::kw_L__FUNCSIG__: IK = PredefinedExpr::LFuncSig; break; // [MS] - case tok::kw___PRETTY_FUNCTION__: IK = PredefinedExpr::PrettyFunction; break; - } - - return BuildPredefinedExpr(Loc, IK); + return BuildPredefinedExpr(Loc, getPredefinedExprKind(Kind)); } ExprResult Sema::ActOnCharacterConstant(const Token &Tok, Scope *UDLScope) { diff --git a/clang/test/Sema/ms_predefined_expr.cpp b/clang/test/Sema/ms_predefined_expr.cpp --- a/clang/test/Sema/ms_predefined_expr.cpp +++ b/clang/test/Sema/ms_predefined_expr.cpp @@ -1,9 +1,156 @@ // RUN: %clang_cc1 %s -fsyntax-only -Wmicrosoft -verify -fms-extensions -void f() { +using size_t = __SIZE_TYPE__; + +// Test array initialization +void array_init() { const char a[] = __FUNCTION__; // expected-warning{{initializing an array from a '__FUNCTION__' predefined identifier is a Microsoft extension}} const char b[] = __FUNCDNAME__; // expected-warning{{initializing an array from a '__FUNCDNAME__' predefined identifier is a Microsoft extension}} const char c[] = __FUNCSIG__; // expected-warning{{initializing an array from a '__FUNCSIG__' predefined identifier is a Microsoft extension}} const char d[] = __func__; // expected-warning{{initializing an array from a '__func__' predefined identifier is a Microsoft extension}} const char e[] = __PRETTY_FUNCTION__; // expected-warning{{initializing an array from a '__PRETTY_FUNCTION__' predefined identifier is a Microsoft extension}} + const wchar_t f[] = L__FUNCTION__; // expected-warning{{initializing an array from a 'L__FUNCTION__' predefined identifier is a Microsoft extension}} + const wchar_t g[] = L__FUNCSIG__; // expected-warning{{initializing an array from a 'L__FUNCSIG__' predefined identifier is a Microsoft extension}} +} + +// Test function local identifiers outside of a function +const char* g_function = __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} +const char* g_function_lconcat = "" __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} +const char* g_function_rconcat = __FUNCTION__ ""; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + +namespace NS +{ + const char* function = __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} + const char* function_lconcat = "" __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + const char* function_rconcat = __FUNCTION__ ""; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + + struct S + { + static constexpr const char* function = __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} + static constexpr const char* function_lconcat = "" __FUNCTION__; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + static constexpr const char* function_rconcat = __FUNCTION__ ""; // expected-warning{{predefined identifier is only valid inside function}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + }; +} + +template +constexpr bool is_same = false; +template +constexpr bool is_same = true; + +template +constexpr bool equal(const T (&a)[N], const T (&b)[N]) { + for (size_t i = 0; i < N; i++) + if (a[i] != b[i]) + return false; + return true; +} + +#define ASSERT_EQ(X, Y) static_assert(equal(X, Y)) +#define ASSERT_EQ_TY(X, Y) static_assert(is_same) + +#define _WIDE(s) L##s +#define WIDE(s) _WIDE(s) + +// Test value +extern "C" void test_value() { + ASSERT_EQ(__FUNCTION__, "test_value"); + ASSERT_EQ(__FUNCSIG__, "void __cdecl test_value(void)"); + + ASSERT_EQ(WIDE(__FUNCTION__), L"test_value"); + ASSERT_EQ(WIDE(__FUNCSIG__), L"void __cdecl test_value(void)"); +} + +namespace PR13206 { + template class A { + public: + void method() { + ASSERT_EQ(L__FUNCTION__, L"method"); + } + }; + + void test_template_value() { + A x; + x.method(); + } +} + +// Test concatenation +extern "C" void test_concat() { + ASSERT_EQ("left_" __FUNCTION__, "left_test_concat"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ("left_" __FUNCDNAME__, "left_test_concat"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCDNAME__' is a Microsoft extension}} + ASSERT_EQ("left " __FUNCSIG__, "left void __cdecl test_concat(void)"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(__FUNCTION__ "_right", "test_concat_right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(__FUNCDNAME__ "_right", "test_concat_right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCDNAME__' is a Microsoft extension}} + ASSERT_EQ(__FUNCSIG__ " right", "void __cdecl test_concat(void) right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ("left_" __FUNCTION__ "_right", "left_test_concat_right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ("left_" __FUNCDNAME__ "_right", "left_test_concat_right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCDNAME__' is a Microsoft extension}} + ASSERT_EQ("left " __FUNCSIG__ " right", "left void __cdecl test_concat(void) right"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(__FUNCTION__ "/" __FUNCSIG__, "test_concat/void __cdecl test_concat(void)"); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} \ + // expected-warning{{string literal concatenation of predefined identifier '__FUNCSIG__' is a Microsoft extension}} +} + +extern "C" void test_wide_concat() { + // test L"" + "" + ASSERT_EQ(L"" __FUNCTION__, L__FUNCTION__); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(L"" __FUNCSIG__, L__FUNCSIG__); // expected-warning{{string literal concatenation of predefined identifier '__FUNCSIG__' is a Microsoft extension}} + + // test Lx + "" + ASSERT_EQ(L__FUNCTION__, L__FUNCTION__ ""); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(L__FUNCSIG__, L__FUNCSIG__ ""); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(L"left_" L__FUNCTION__, L"left_test_wide_concat"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(L"left " L__FUNCSIG__, L"left void __cdecl test_wide_concat(void)"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(L__FUNCTION__ L"_right", L"test_wide_concat_right"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(L__FUNCSIG__ L" right", L"void __cdecl test_wide_concat(void) right"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(L"left_" L__FUNCTION__ L"_right", L"left_test_wide_concat_right"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ(L"left " L__FUNCSIG__ L" right", L"left void __cdecl test_wide_concat(void) right"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCSIG__' is a Microsoft extension}} + + ASSERT_EQ(L__FUNCTION__ L"/" L__FUNCSIG__, L"test_wide_concat/void __cdecl test_wide_concat(void)"); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} \ + // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCSIG__' is a Microsoft extension}} +} + +void test_encoding() { + ASSERT_EQ_TY("" __FUNCTION__, ""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(L"" __FUNCTION__, L""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(L"" L__FUNCTION__, L""); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY("" L__FUNCTION__, L""); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(u8"" __FUNCTION__, u8""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(u"" __FUNCTION__, u""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(U"" __FUNCTION__, U""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + // + ASSERT_EQ_TY(__FUNCTION__ L"", L""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(__FUNCTION__ u8"", u8""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(__FUNCTION__ u"", u""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} + ASSERT_EQ_TY(__FUNCTION__ U"", U""); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} +} + +template +void unused(T); + +void test_invalid_encoding() { + unused(u8"" L__FUNCTION__); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} \ + // expected-error{{unsupported non-standard concatenation of string literals}} + unused(u"" L__FUNCTION__); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} \ + // expected-error{{unsupported non-standard concatenation of string literals}} + unused(U"" L__FUNCTION__); // expected-warning{{string literal concatenation of predefined identifier 'L__FUNCTION__' is a Microsoft extension}} \ + // expected-error{{unsupported non-standard concatenation of string literals}} +} + +constexpr size_t operator""_len(const char*, size_t len) { + return len; +} + +void test_udliteral() { + static_assert(__FUNCTION__ ""_len == 14); // expected-warning{{string literal concatenation of predefined identifier '__FUNCTION__' is a Microsoft extension}} } diff --git a/clang/test/Sema/ms_wide_predefined_expr.cpp b/clang/test/Sema/ms_wide_predefined_expr.cpp deleted file mode 100644 --- a/clang/test/Sema/ms_wide_predefined_expr.cpp +++ /dev/null @@ -1,28 +0,0 @@ -// RUN: %clang_cc1 %s -fsyntax-only -Wno-unused-value -Wmicrosoft -verify -fms-extensions -// expected-no-diagnostics - -// Wide character predefined identifiers -#define _STR2WSTR(str) L##str -#define STR2WSTR(str) _STR2WSTR(str) -void abcdefghi12(void) { - const wchar_t (*ss)[12] = &STR2WSTR(__FUNCTION__); - static int arr[sizeof(STR2WSTR(__FUNCTION__))==12*sizeof(wchar_t) ? 1 : -1]; - const wchar_t (*ss2)[31] = &STR2WSTR(__FUNCSIG__); - static int arr2[sizeof(STR2WSTR(__FUNCSIG__))==31*sizeof(wchar_t) ? 1 : -1]; -} - -namespace PR13206 { -void foo(const wchar_t *); - -template class A { -public: - void method() { - foo(L__FUNCTION__); - } -}; - -void bar() { - A x; - x.method(); -} -}