[clang] [Clang] Fixed a crash when instantiating an invalid out-of-line static data member definition in a local class (PR #196772)

via cfe-commits cfe-commits at lists.llvm.org
Sun May 10 08:49:01 PDT 2026


https://github.com/TPPPP72 updated https://github.com/llvm/llvm-project/pull/196772

>From c03519a4223c2a46dd6e68cbf061205fa832b499 Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 09:53:16 +0800
Subject: [PATCH 1/6] temp

---
 clang/lib/Sema/SemaDecl.cpp                    | 1 +
 clang/lib/Sema/SemaTemplateInstantiateDecl.cpp | 8 ++++++++
 2 files changed, 9 insertions(+)

diff --git a/clang/lib/Sema/SemaDecl.cpp b/clang/lib/Sema/SemaDecl.cpp
index a9a4cb89d115f..d9ba3b30dcca9 100644
--- a/clang/lib/Sema/SemaDecl.cpp
+++ b/clang/lib/Sema/SemaDecl.cpp
@@ -7948,6 +7948,7 @@ NamedDecl *Sema::ActOnVariableDeclarator(
             Diag(D.getIdentifierLoc(),
                  diag::err_static_data_member_not_allowed_in_local_class)
                 << Name << RD->getDeclName() << RD->getTagKind();
+            Invalid = true;
           } else if (AnonStruct) {
             // C++ [class.static.data]p4: Unnamed classes and classes contained
             // directly or indirectly within unnamed classes shall not contain
diff --git a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
index c9bc613a7c4ea..2a4f459dbe40f 100644
--- a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
+++ b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
@@ -1799,6 +1799,14 @@ Decl *TemplateDeclInstantiator::VisitVarDecl(VarDecl *D,
   if (SemaRef.getLangOpts().OpenCL)
     SemaRef.deduceOpenCLAddressSpace(Var);
 
+  if (D->getQualifierLoc()) {
+    if (D->getLexicalDeclContext()->isDependentContext() &&
+        DC->isFunctionOrMethod()) {
+      Var->setInvalidDecl();
+      return nullptr;
+    }
+  }
+
   // Substitute the nested name specifier, if any.
   if (SubstQualifier(D, Var))
     return nullptr;

>From b8803bc218f49f06ac7bc1b8dfb6bdc78a857169 Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 10:44:28 +0800
Subject: [PATCH 2/6] [Clang] Fixed a crash when instantiating an invalid
 out-of-line static data member definition in a local class

---
 clang/docs/ReleaseNotes.rst          |  2 +-
 clang/test/SemaTemplate/gh176152.cpp | 13 +++++++++++++
 2 files changed, 14 insertions(+), 1 deletion(-)
 create mode 100644 clang/test/SemaTemplate/gh176152.cpp

diff --git a/clang/docs/ReleaseNotes.rst b/clang/docs/ReleaseNotes.rst
index c17143e3c0398..06adba419b192 100644
--- a/clang/docs/ReleaseNotes.rst
+++ b/clang/docs/ReleaseNotes.rst
@@ -581,7 +581,7 @@ Bug Fixes to C++ Support
 - Fixed a crash on ``typeid`` of incomplete local types during template instantiation. (#GH63242), (#GH176397)
 - Fixed a crash when an immediate-invoked ``consteval`` lambda is used as an invalid initializer. (#GH185270)
 - Fixed an assertion failure when using a global destructor with a target with a non-default program address space. (#GH186484)
-
+- Fixed a crash when instantiating an invalid out-of-line static data member definition in a local class. (#GH176152)
 - Inherited constructors in ``dllexport`` classes are now exported for ABI-compatible cases, matching
   MSVC behavior. Constructors with variadic arguments or callee-cleanup parameters are not yet supported
   and produce a warning. (#GH162640)
diff --git a/clang/test/SemaTemplate/gh176152.cpp b/clang/test/SemaTemplate/gh176152.cpp
new file mode 100644
index 0000000000000..feae5d0f0a825
--- /dev/null
+++ b/clang/test/SemaTemplate/gh176152.cpp
@@ -0,0 +1,13 @@
+// RUN: %clang_cc1 -fsyntax-only -verify %s
+
+template <class T> int f(T) {
+  struct A {
+    static int B;
+    // expected-error at -1 {{static data member 'staticField' not allowed in local struct 'MyClass'}}
+  };
+  int A::B;
+  int A::B = 1;
+  return 0;
+}
+
+int x = f(0);
\ No newline at end of file

>From 6f394a5df991acb2f62d653201af09f93f790a80 Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 11:06:03 +0800
Subject: [PATCH 3/6] Add a blank line

---
 clang/test/SemaTemplate/gh176152.cpp | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/clang/test/SemaTemplate/gh176152.cpp b/clang/test/SemaTemplate/gh176152.cpp
index feae5d0f0a825..27cb1c53dcf16 100644
--- a/clang/test/SemaTemplate/gh176152.cpp
+++ b/clang/test/SemaTemplate/gh176152.cpp
@@ -10,4 +10,4 @@ template <class T> int f(T) {
   return 0;
 }
 
-int x = f(0);
\ No newline at end of file
+int x = f(0);

>From 2eae3d9f5f7bbdad964b76a24e462695d269121a Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 12:03:20 +0800
Subject: [PATCH 4/6] fix test

---
 clang/test/SemaTemplate/gh176152.cpp | 7 +++----
 1 file changed, 3 insertions(+), 4 deletions(-)

diff --git a/clang/test/SemaTemplate/gh176152.cpp b/clang/test/SemaTemplate/gh176152.cpp
index 27cb1c53dcf16..288b9d645faec 100644
--- a/clang/test/SemaTemplate/gh176152.cpp
+++ b/clang/test/SemaTemplate/gh176152.cpp
@@ -2,11 +2,10 @@
 
 template <class T> int f(T) {
   struct A {
-    static int B;
-    // expected-error at -1 {{static data member 'staticField' not allowed in local struct 'MyClass'}}
+    static int B; // expected-error {{static data member 'B' not allowed in local struct 'A'}}
   };
-  int A::B;
-  int A::B = 1;
+  int A::B; // expected-note {{previous definition is here}}
+  int A::B = 1; // expected-error {{redefinition of 'B'}}
   return 0;
 }
 

>From 55bf2b1d47c7f73a3725350c26b2a84cda31dc21 Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 23:45:54 +0800
Subject: [PATCH 5/6] remove check in VisitVarDecl

---
 clang/lib/Sema/SemaDecl.cpp                    | 2 ++
 clang/lib/Sema/SemaTemplateInstantiateDecl.cpp | 8 --------
 2 files changed, 2 insertions(+), 8 deletions(-)

diff --git a/clang/lib/Sema/SemaDecl.cpp b/clang/lib/Sema/SemaDecl.cpp
index d9ba3b30dcca9..eedbaf425666b 100644
--- a/clang/lib/Sema/SemaDecl.cpp
+++ b/clang/lib/Sema/SemaDecl.cpp
@@ -4898,6 +4898,8 @@ void Sema::MergeVarDecl(VarDecl *New, LookupResult &Previous) {
       VarDecl *Def = Old->getDefinition();
       if (Def && checkVarDeclRedefinition(Def, New))
         return;
+      if(Old->isInvalidDecl())
+        New->setInvalidDecl();
     }
   } else {
     // C++ may not have a tentative definition rule, but it has a different
diff --git a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
index 2a4f459dbe40f..c9bc613a7c4ea 100644
--- a/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
+++ b/clang/lib/Sema/SemaTemplateInstantiateDecl.cpp
@@ -1799,14 +1799,6 @@ Decl *TemplateDeclInstantiator::VisitVarDecl(VarDecl *D,
   if (SemaRef.getLangOpts().OpenCL)
     SemaRef.deduceOpenCLAddressSpace(Var);
 
-  if (D->getQualifierLoc()) {
-    if (D->getLexicalDeclContext()->isDependentContext() &&
-        DC->isFunctionOrMethod()) {
-      Var->setInvalidDecl();
-      return nullptr;
-    }
-  }
-
   // Substitute the nested name specifier, if any.
   if (SubstQualifier(D, Var))
     return nullptr;

>From b5a6045096b40044c0bf62013f7d2c4a217ac7ad Mon Sep 17 00:00:00 2001
From: TPPPP <TPPPP72 at outlook.com>
Date: Sun, 10 May 2026 23:48:31 +0800
Subject: [PATCH 6/6] format

---
 clang/lib/Sema/SemaDecl.cpp | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/clang/lib/Sema/SemaDecl.cpp b/clang/lib/Sema/SemaDecl.cpp
index eedbaf425666b..3c5f2fa4a7389 100644
--- a/clang/lib/Sema/SemaDecl.cpp
+++ b/clang/lib/Sema/SemaDecl.cpp
@@ -4898,7 +4898,7 @@ void Sema::MergeVarDecl(VarDecl *New, LookupResult &Previous) {
       VarDecl *Def = Old->getDefinition();
       if (Def && checkVarDeclRedefinition(Def, New))
         return;
-      if(Old->isInvalidDecl())
+      if (Old->isInvalidDecl())
         New->setInvalidDecl();
     }
   } else {



More information about the cfe-commits mailing list