https://github.com/Michael137 updated 
https://github.com/llvm/llvm-project/pull/147283

>From be3eb2431b0649ce2730a4cf832dc474e262f49e Mon Sep 17 00:00:00 2001
From: Michael Buch <michaelbuc...@gmail.com>
Date: Mon, 7 Jul 2025 12:40:37 +0100
Subject: [PATCH 1/3] [lldb][test] Fix libstdc++ std::variant formatter for
 empty variant

Instead of using the byte-size to make a guess at what the
`std::variant_npos` value is, just look it up in debug-info.

Unblocks https://github.com/llvm/llvm-project/pull/147253
---
 lldb/examples/synthetic/gnu_libstdcpp.py            | 13 ++++---------
 .../variant/TestDataFormatterLibStdcxxVariant.py    |  5 -----
 .../data-formatter-stl/libstdcpp/variant/main.cpp   |  1 +
 3 files changed, 5 insertions(+), 14 deletions(-)

diff --git a/lldb/examples/synthetic/gnu_libstdcpp.py 
b/lldb/examples/synthetic/gnu_libstdcpp.py
index 20b9488af5597..96b11322db775 100644
--- a/lldb/examples/synthetic/gnu_libstdcpp.py
+++ b/lldb/examples/synthetic/gnu_libstdcpp.py
@@ -889,17 +889,12 @@ def VariantSummaryProvider(valobj, dict):
     if not (index_obj and index_obj.IsValid() and data_obj and 
data_obj.IsValid()):
         return "<Can't find _M_index or _M_u>"
 
-    def get_variant_npos_value(index_byte_size):
-        if index_byte_size == 1:
-            return 0xFF
-        elif index_byte_size == 2:
-            return 0xFFFF
-        else:
-            return 0xFFFFFFFF
+    npos = valobj.GetTarget().FindFirstGlobalVariable("std::variant_npos")
+    if not npos:
+        return "<Can't find std::variant_npos sentinel>"
 
-    npos_value = get_variant_npos_value(index_obj.GetByteSize())
     index = index_obj.GetValueAsUnsigned(0)
-    if index == npos_value:
+    if index == npos.GetValueAsUnsigned(0):
         return " No Value"
 
     # Strip references and typedefs.
diff --git 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
index 394e221809f7c..c3325c9e73cb9 100644
--- 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
+++ 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
@@ -2,7 +2,6 @@
 Test lldb data formatter for LibStdC++ std::variant.
 """
 
-
 import lldb
 from lldbsuite.test.decorators import *
 from lldbsuite.test.lldbtest import *
@@ -62,9 +61,6 @@ def test_with_run_command(self):
             "frame variable v3",
             substrs=["v3 =  Active Type = char  {", "Value = 'A'", "}"],
         )
-        """
-        TODO: temporarily disable No Value tests as they seem to fail on 
ubuntu/debian
-        bots. Pending reproduce and investigation.
 
         self.expect("frame variable v_no_value", substrs=["v_no_value =  No 
Value"])
 
@@ -72,7 +68,6 @@ def test_with_run_command(self):
             "frame variable v_many_types_no_value",
             substrs=["v_many_types_no_value =  No Value"],
         )
-        """
 
     @add_test_categories(["libstdcxx"])
     def test_invalid_variant_index(self):
diff --git 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
index 36e0f74f831f8..acb07d5700b8a 100644
--- 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
+++ 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
@@ -49,6 +49,7 @@ int main() {
   v1 = 12; // v contains int
   v1_typedef = v1;
   v_v1 = v1;
+  v1_typedef = v1;
   int i = std::get<int>(v1);
   printf("%d\n", i); // break here
 

>From 59518c7f14a6beeb29ebdc8b33eeede9176418b3 Mon Sep 17 00:00:00 2001
From: Michael Buch <michaelbuc...@gmail.com>
Date: Mon, 7 Jul 2025 12:47:28 +0100
Subject: [PATCH 2/3] fixup! revert redundant test change

---
 .../data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp | 1 -
 1 file changed, 1 deletion(-)

diff --git 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
index acb07d5700b8a..36e0f74f831f8 100644
--- 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
+++ 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
@@ -49,7 +49,6 @@ int main() {
   v1 = 12; // v contains int
   v1_typedef = v1;
   v_v1 = v1;
-  v1_typedef = v1;
   int i = std::get<int>(v1);
   printf("%d\n", i); // break here
 

>From 788661c1c945b6a2b5190812a1b232f335096388 Mon Sep 17 00:00:00 2001
From: Michael Buch <michaelbuc...@gmail.com>
Date: Mon, 7 Jul 2025 13:52:44 +0100
Subject: [PATCH 3/3] fixup! fix test; revert changes to formatter

---
 lldb/examples/synthetic/gnu_libstdcpp.py      | 13 +++++++---
 .../TestDataFormatterLibStdcxxVariant.py      |  6 ++---
 .../libstdcpp/variant/main.cpp                | 25 +++++++++++++------
 3 files changed, 29 insertions(+), 15 deletions(-)

diff --git a/lldb/examples/synthetic/gnu_libstdcpp.py 
b/lldb/examples/synthetic/gnu_libstdcpp.py
index 96b11322db775..20b9488af5597 100644
--- a/lldb/examples/synthetic/gnu_libstdcpp.py
+++ b/lldb/examples/synthetic/gnu_libstdcpp.py
@@ -889,12 +889,17 @@ def VariantSummaryProvider(valobj, dict):
     if not (index_obj and index_obj.IsValid() and data_obj and 
data_obj.IsValid()):
         return "<Can't find _M_index or _M_u>"
 
-    npos = valobj.GetTarget().FindFirstGlobalVariable("std::variant_npos")
-    if not npos:
-        return "<Can't find std::variant_npos sentinel>"
+    def get_variant_npos_value(index_byte_size):
+        if index_byte_size == 1:
+            return 0xFF
+        elif index_byte_size == 2:
+            return 0xFFFF
+        else:
+            return 0xFFFFFFFF
 
+    npos_value = get_variant_npos_value(index_obj.GetByteSize())
     index = index_obj.GetValueAsUnsigned(0)
-    if index == npos.GetValueAsUnsigned(0):
+    if index == npos_value:
         return " No Value"
 
     # Strip references and typedefs.
diff --git 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
index c3325c9e73cb9..dae9b24fbbcfe 100644
--- 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
+++ 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/TestDataFormatterLibStdcxxVariant.py
@@ -62,11 +62,11 @@ def test_with_run_command(self):
             substrs=["v3 =  Active Type = char  {", "Value = 'A'", "}"],
         )
 
-        self.expect("frame variable v_no_value", substrs=["v_no_value =  No 
Value"])
+        self.expect("frame variable v_valueless", substrs=["v_valueless =  No 
Value"])
 
         self.expect(
-            "frame variable v_many_types_no_value",
-            substrs=["v_many_types_no_value =  No Value"],
+            "frame variable v_many_types_valueless",
+            substrs=["v_many_types_valueless =  No Value"],
         )
 
     @add_test_categories(["libstdcxx"])
diff --git 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
index 36e0f74f831f8..235928264add1 100644
--- 
a/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
+++ 
b/lldb/test/API/functionalities/data-formatter/data-formatter-stl/libstdcpp/variant/main.cpp
@@ -4,7 +4,9 @@
 #include <vector>
 
 struct S {
-  operator int() { throw 42; }
+  S() = default;
+  S(S &&) { throw 42; }
+  S &operator=(S &&) = default;
 };
 
 int main() {
@@ -21,7 +23,7 @@ int main() {
   std::variant<int, double, char> v2;
   std::variant<int, double, char> v3;
   std::variant<std::variant<int, double, char>> v_v1;
-  std::variant<int, double, char> v_no_value;
+  std::variant<int, char, S> v_valueless = 5;
   // The next variant has many types, meaning the type index does not fit in
   // a byte and must be `unsigned short` instead of `unsigned char` when
   // using the unstable libc++ ABI. With stable libc++ ABI, the type index
@@ -43,8 +45,11 @@ int main() {
       int, int, int, int, int, int, int, int, int, int, int, int, int, int, 
int,
       int, int, int, int, int, int, int, int, int, int, int, int, int, int, 
int,
       int, int, int, int, int, int, int, int, int, int, int, int, int, int, 
int,
-      int, int, int, int, int, int, int, int, int, int, int, int>
-      v_many_types_no_value;
+      int, int, int, int, int, int, int, int, int, int, int, int, S>
+      v_many_types_valueless;
+
+  v_valueless = 5;
+  v_many_types_valueless.emplace<0>(10);
 
   v1 = 12; // v contains int
   v1_typedef = v1;
@@ -67,18 +72,22 @@ int main() {
   printf("%f\n", d); // break here
 
   try {
-    v_no_value.emplace<0>(S());
+    // Exception in type-changing move-assignment is guaranteed to put
+    // std::variant into a valueless state.
+    v_valueless = S();
   } catch (...) {
   }
 
-  printf("%zu\n", v_no_value.index());
+  printf("%d\n", v_valueless.valueless_by_exception());
 
   try {
-    v_many_types_no_value.emplace<0>(S());
+    // Exception in move-assignment is guaranteed to put std::variant into a
+    // valueless state.
+    v_many_types_valueless = S();
   } catch (...) {
   }
 
-  printf("%zu\n", v_many_types_no_value.index());
+  printf("%d\n", v_many_types_valueless.valueless_by_exception());
 
   return 0; // break here
 }

_______________________________________________
lldb-commits mailing list
lldb-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits

Reply via email to