This patch adds support for template arguments of `clang::TemplateArgument::ArgKind::StructuralValue` kind (added in https://github.com/llvm/llvm-project/pull/78041). These are used for non-type template parameters such as floating point constants. When LLDB created `clang::NonTypeTemplateParmDecl`s, it previously assumed integral values, this patch accounts for structural values too. Anywhere LLDB assumed a `DW_TAG_template_value_parameter` was `Integral`, it will now also check for `StructuralValue`, and will unpack the `TemplateArgument` value and type accordingly. We can rely on the fact that any `TemplateArgument` of `StructuralValue` kind that the `DWARFASTParserClang` creates will have a valid value, because it gets those from `DW_AT_const_value`.
106 lines
4.8 KiB
Python
106 lines
4.8 KiB
Python
import lldb
|
|
from lldbsuite.test.decorators import *
|
|
from lldbsuite.test.lldbtest import *
|
|
from lldbsuite.test import lldbutil
|
|
|
|
|
|
class TestCase(TestBase):
|
|
@no_debug_info_test
|
|
@skipIf(compiler="clang", compiler_version=["<", "20.0"])
|
|
def test(self):
|
|
self.build()
|
|
target = self.dbg.CreateTarget(self.getBuildArtifact("a.out"))
|
|
|
|
value = self.expect_expr("temp1", result_type="C<int, 2>")
|
|
template_type = value.GetType()
|
|
self.assertEqual(template_type.GetNumberOfTemplateArguments(), 2)
|
|
|
|
# Check a type argument.
|
|
self.assertEqual(
|
|
template_type.GetTemplateArgumentKind(0), lldb.eTemplateArgumentKindType
|
|
)
|
|
self.assertEqual(template_type.GetTemplateArgumentType(0).GetName(), "int")
|
|
|
|
# Check a integral argument.
|
|
self.assertEqual(
|
|
template_type.GetTemplateArgumentKind(1), lldb.eTemplateArgumentKindIntegral
|
|
)
|
|
self.assertEqual(
|
|
template_type.GetTemplateArgumentType(1).GetName(), "unsigned int"
|
|
)
|
|
|
|
# Template parameter isn't a NTTP.
|
|
self.assertFalse(template_type.GetTemplateArgumentValue(target, 0))
|
|
|
|
# Template parameter index out-of-bounds.
|
|
self.assertFalse(template_type.GetTemplateArgumentValue(target, 2))
|
|
|
|
# Template parameter is a NTTP.
|
|
param_val = template_type.GetTemplateArgumentValue(target, 1)
|
|
self.assertEqual(param_val.GetTypeName(), "unsigned int")
|
|
self.assertEqual(param_val.GetValueAsUnsigned(), 2)
|
|
|
|
# Try to get an invalid template argument.
|
|
self.assertEqual(
|
|
template_type.GetTemplateArgumentKind(2), lldb.eTemplateArgumentKindNull
|
|
)
|
|
self.assertEqual(template_type.GetTemplateArgumentType(2).GetName(), "")
|
|
|
|
value = self.expect_expr("temp2", result_type="Foo<short, -2>")
|
|
|
|
# Can't get template parameter value with invalid target.
|
|
self.assertFalse(value.GetType().GetTemplateArgumentValue(lldb.SBTarget(), 1))
|
|
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertTrue(template_param_value)
|
|
self.assertEqual(template_param_value.GetTypeName(), "short")
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), -2)
|
|
|
|
value = self.expect_expr("temp3", result_type="Foo<char, 'v'>")
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertTrue(template_param_value)
|
|
self.assertEqual(template_param_value.GetTypeName(), "char")
|
|
self.assertEqual(chr(template_param_value.GetValueAsSigned()), "v")
|
|
|
|
value = self.expect_expr("temp4", result_type="Foo<float, 2.000000e+00>")
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertEqual(template_param_value.GetTypeName(), "float")
|
|
# FIXME: this should return a float
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), 2)
|
|
|
|
value = self.expect_expr("temp5", result_type="Foo<double, -2.505000e+02>")
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertEqual(template_param_value.GetTypeName(), "double")
|
|
# FIXME: this should return a float
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), -250)
|
|
|
|
# FIXME: type should be Foo<int *, &temp1.member>
|
|
value = self.expect_expr("temp6", result_type="Foo<int *, int *>")
|
|
self.assertFalse(value.GetType().GetTemplateArgumentValue(target, 1))
|
|
|
|
# FIXME: support wider range of floating point types
|
|
value = self.expect_expr("temp7", result_type="Foo<__fp16, __fp16>")
|
|
self.assertFalse(value.GetType().GetTemplateArgumentValue(target, 1))
|
|
|
|
value = self.expect_expr("temp8", result_type="Foo<__fp16, __fp16>")
|
|
self.assertFalse(value.GetType().GetTemplateArgumentValue(target, 1))
|
|
|
|
value = self.expect_expr("temp9", result_type="Bar<double, 1.200000e+00>")
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertEqual(template_param_value.GetTypeName(), "double")
|
|
# FIXME: this should return a float
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), 1)
|
|
|
|
value = self.expect_expr(
|
|
"temp10", result_type="Bar<float, 1.000000e+00, 2.000000e+00>"
|
|
)
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 1)
|
|
self.assertEqual(template_param_value.GetTypeName(), "float")
|
|
# FIXME: this should return a float
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), 1)
|
|
|
|
template_param_value = value.GetType().GetTemplateArgumentValue(target, 2)
|
|
self.assertEqual(template_param_value.GetTypeName(), "float")
|
|
# FIXME: this should return a float
|
|
self.assertEqual(template_param_value.GetValueAsSigned(), 2)
|