"""
Test SBType and SBTypeList API.
"""
import lldb
from lldbsuite.test.decorators import *
from lldbsuite.test.lldbtest import *
from lldbsuite.test import lldbutil
class TypeAndTypeListTestCase(TestBase):
def setUp(self):
# Call super's setUp().
TestBase.setUp(self)
# We'll use the test method name as the exe_name.
self.exe_name = self.testMethodName
# Find the line number to break at.
self.source = "main.cpp"
self.line = line_number(self.source, "// Break at this line")
def _find_nested_type_in_Pointer_template_arg(self, pointer_type):
self.assertTrue(pointer_type)
self.DebugSBType(pointer_type)
pointer_info_type = pointer_type.template_args[1]
self.assertTrue(pointer_info_type)
self.DebugSBType(pointer_info_type)
pointer_masks1_type = pointer_info_type.FindDirectNestedType("Masks1")
self.assertTrue(pointer_masks1_type)
self.DebugSBType(pointer_masks1_type)
pointer_masks2_type = pointer_info_type.FindDirectNestedType("Masks2")
self.assertTrue(pointer_masks2_type)
self.DebugSBType(pointer_masks2_type)
def _find_static_field_in_Task_pointer(self, task_pointer):
self.assertTrue(task_pointer)
self.DebugSBType(task_pointer)
task_type = task_pointer.GetPointeeType()
self.assertTrue(task_type)
self.DebugSBType(task_type)
static_constexpr_field = task_type.GetStaticFieldWithName(
"static_constexpr_field"
)
self.assertTrue(static_constexpr_field)
self.assertEqual(static_constexpr_field.GetName(), "static_constexpr_field")
self.assertEqual(static_constexpr_field.GetType().GetName(), "const long")
value = static_constexpr_field.GetConstantValue(self.target())
self.DebugSBValue(value)
self.assertEqual(value.GetValueAsSigned(), 47)
static_constexpr_bool_field = task_type.GetStaticFieldWithName(
"static_constexpr_bool_field"
)
self.assertTrue(static_constexpr_bool_field)
self.assertEqual(
static_constexpr_bool_field.GetName(), "static_constexpr_bool_field"
)
self.assertEqual(static_constexpr_bool_field.GetType().GetName(), "const bool")
value = static_constexpr_bool_field.GetConstantValue(self.target())
self.DebugSBValue(value)
self.assertEqual(value.GetValueAsUnsigned(), 1)
static_mutable_field = task_type.GetStaticFieldWithName("static_mutable_field")
self.assertTrue(static_mutable_field)
self.assertEqual(static_mutable_field.GetName(), "static_mutable_field")
self.assertEqual(static_mutable_field.GetType().GetName(), "int")
self.assertFalse(static_mutable_field.GetConstantValue(self.target()))
@skipIf(compiler="clang", compiler_version=["<", "17.0"])
def test(self):
"""Exercise SBType and SBTypeList API."""
d = {"EXE": self.exe_name}
self.build(dictionary=d)
self.setTearDownCleanup(dictionary=d)
exe = self.getBuildArtifact(self.exe_name)
# Create a target by the debugger.
target = self.dbg.CreateTarget(exe)
self.assertTrue(target, VALID_TARGET)
# Create the breakpoint inside function 'main'.
breakpoint = target.BreakpointCreateByLocation(self.source, self.line)
self.assertTrue(breakpoint, VALID_BREAKPOINT)
# Now launch the process, and do not stop at entry point.
process = target.LaunchSimple(None, None, self.get_process_working_directory())
self.assertTrue(process, PROCESS_IS_VALID)
# Get Frame #0.
self.assertState(process.GetState(), lldb.eStateStopped)
thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint)
self.assertTrue(
thread.IsValid(),
"There should be a thread stopped due to breakpoint condition",
)
frame0 = thread.GetFrameAtIndex(0)
# Get the type 'Task'.
type_list = target.FindTypes("Task")
if self.TraceOn():
print(
"Size of type_list from target.FindTypes('Task') query: %d"
% type_list.GetSize()
)
# a second Task make be scared up by the Objective-C runtime
self.assertGreaterEqual(len(type_list), 1)
for type in type_list:
self.assertTrue(type)
self.DebugSBType(type)
self.assertFalse(type.IsAnonymousType(), "Task is not anonymous")
self.assertTrue(type.IsAggregateType(), "Task is aggregate")
for field in type.fields:
if field.name == "type":
for enum_member in field.type.enum_members:
self.assertTrue(enum_member)
self.DebugSBType(enum_member.type)
elif field.name == "my_type_is_nameless":
self.assertFalse(
field.type.IsAnonymousType(),
"my_type_is_nameless is not an anonymous type",
)
self.assertTrue(field.type.IsAggregateType())
elif field.name == "my_type_is_named":
self.assertFalse(
field.type.IsAnonymousType(),
"my_type_is_named has a named type",
)
self.assertTrue(field.type.IsAggregateType())
elif field.name is None:
self.assertTrue(
field.type.IsAnonymousType(), "Nameless type is not anonymous"
)
# Pass an empty string. LLDB should not crash. :-)
fuzz_types = target.FindTypes(None)
fuzz_type = target.FindFirstType(None)
# Now use the SBTarget.FindFirstType() API to find 'Task'.
task_type = target.FindFirstType("Task")
self.assertTrue(task_type)
self.DebugSBType(task_type)
# Get the reference type of 'Task', just for fun.
task_ref_type = task_type.GetReferenceType()
self.assertTrue(task_ref_type)
self.DebugSBType(task_ref_type)
# Get the pointer type of 'Task', which is the same as task_head's
# type.
task_pointer_type = task_type.GetPointerType()
self.assertTrue(task_pointer_type)
self.DebugSBType(task_pointer_type)
# Get variable 'task_head'.
task_head = frame0.FindVariable("task_head")
self.assertTrue(task_head, VALID_VARIABLE)
self.DebugSBValue(task_head)
task_head_type = task_head.GetType()
self.DebugSBType(task_head_type)
self.assertTrue(task_head_type.IsPointerType())
self.assertFalse(task_head_type.IsAggregateType())
self.assertEqual(task_head_type, task_pointer_type)
# Get the pointee type of 'task_head'.
task_head_pointee_type = task_head_type.GetPointeeType()
self.DebugSBType(task_head_pointee_type)
self.assertEqual(task_type, task_head_pointee_type)
# Check whether we can find a directly nested type by name
name_type = task_type.FindDirectNestedType("name")
self.assertTrue(name_type)
self.DebugSBType(name_type)
enum_type = task_type.FindDirectNestedType("E")
self.assertTrue(enum_type)
self.DebugSBType(enum_type)
union_type = task_type.FindDirectNestedType("U")
self.assertTrue(union_type)
self.DebugSBType(union_type)
# Check that we don't find indirectly nested types
self.assertEqual(enum_type.size, 1)
invalid_type = task_type.FindDirectNestedType("E2")
self.assertFalse(invalid_type)
# Check that FindDirectNestedType handles types without DeclContext
# and other errorneous inputs
task_ptr_type = task_type.GetPointerType()
invalid_type = task_ptr_type.FindDirectNestedType("name")
self.assertFalse(invalid_type)
invalid_type = task_type.FindDirectNestedType("")
self.assertFalse(invalid_type)
invalid_type = task_type.FindDirectNestedType(None)
self.assertFalse(invalid_type)
# Check that FindDirectNestedType works with types from module and
# expression ASTs.
self._find_nested_type_in_Pointer_template_arg(
frame0.FindVariable("pointer").GetType()
)
self._find_nested_type_in_Pointer_template_arg(
frame0.EvaluateExpression("pointer").GetType()
)
self._find_static_field_in_Task_pointer(
frame0.FindVariable("task_head").GetType()
)
self._find_static_field_in_Task_pointer(
frame0.EvaluateExpression("task_head").GetType()
)
# We'll now get the child member 'id' from 'task_head'.
id = task_head.GetChildMemberWithName("id")
self.DebugSBValue(id)
id_type = id.GetType()
self.DebugSBType(id_type)
self.assertFalse(id_type.IsAggregateType())
# SBType.GetBasicType() takes an enum 'BasicType'
# (lldb-enumerations.h).
int_type = id_type.GetBasicType(lldb.eBasicTypeInt)
self.assertEqual(id_type, int_type)
# Find 'myint_arr' and check the array element type.
myint_arr = frame0.FindVariable("myint_arr")
self.assertTrue(myint_arr, VALID_VARIABLE)
self.DebugSBValue(myint_arr)
myint_arr_type = myint_arr.GetType()
self.DebugSBType(myint_arr_type)
self.assertTrue(myint_arr_type.IsArrayType())
self.assertTrue(myint_arr_type.IsAggregateType())
myint_arr_element_type = myint_arr_type.GetArrayElementType()
self.DebugSBType(myint_arr_element_type)
myint_type = target.FindFirstType("myint")
self.DebugSBType(myint_type)
self.assertEqual(myint_arr_element_type, myint_type)
# Test enum methods. Requires DW_AT_enum_class which was added in Dwarf 4.
if configuration.dwarf_version >= 4:
enum_type = target.FindFirstType("EnumType")
self.assertTrue(enum_type)
self.DebugSBType(enum_type)
self.assertFalse(enum_type.IsScopedEnumerationType())
self.assertFalse(enum_type.IsAggregateType())
scoped_enum_type = target.FindFirstType("ScopedEnumType")
self.assertTrue(scoped_enum_type)
self.DebugSBType(scoped_enum_type)
self.assertTrue(scoped_enum_type.IsScopedEnumerationType())
self.assertFalse(scoped_enum_type.IsAggregateType())
int_scoped_enum_type = scoped_enum_type.GetEnumerationIntegerType()
self.assertTrue(int_scoped_enum_type)
self.DebugSBType(int_scoped_enum_type)
self.assertEqual(int_scoped_enum_type.GetName(), "int")
enum_uchar = target.FindFirstType("EnumUChar")
self.assertTrue(enum_uchar)
self.DebugSBType(enum_uchar)
int_enum_uchar = enum_uchar.GetEnumerationIntegerType()
self.assertTrue(int_enum_uchar)
self.DebugSBType(int_enum_uchar)
self.assertEqual(int_enum_uchar.GetName(), "unsigned char")
def test_nested_typedef(self):
"""Exercise FindDirectNestedType for typedefs."""
self.build()
target = self.dbg.CreateTarget(self.getBuildArtifact())
self.assertTrue(target)
with_nested_typedef = target.FindFirstType("WithNestedTypedef")
self.assertTrue(with_nested_typedef)
# This is necessary to work around #91186
self.assertTrue(target.FindFirstGlobalVariable("typedefed_value").GetType())
the_typedef = with_nested_typedef.FindDirectNestedType("TheTypedef")
self.assertTrue(the_typedef)
self.assertEqual(the_typedef.GetTypedefedType().GetName(), "int")
def test_GetByteAlign(self):
"""Exercise SBType::GetByteAlign"""
self.build()
spec = lldb.SBModuleSpec()
spec.SetFileSpec(lldb.SBFileSpec(self.getBuildArtifact()))
module = lldb.SBModule(spec)
self.assertTrue(module)
# Invalid types should not crash.
self.assertEqual(lldb.SBType().GetByteAlign(), 0)
# Try a type with natural alignment.
void_ptr = module.GetBasicType(lldb.eBasicTypeVoid).GetPointerType()
self.assertTrue(void_ptr)
# Not exactly guaranteed by the spec, but should be true everywhere we
# care about.
self.assertEqual(void_ptr.GetByteSize(), void_ptr.GetByteAlign())
# And an over-aligned type.
self.assertEqual(module.FindFirstType("OverAlignedStruct").GetByteAlign(), 128)