199451b44SJordan Rupprecht""" 299451b44SJordan RupprechtTest display and Python APIs on file and class static variables. 399451b44SJordan Rupprecht""" 499451b44SJordan Rupprecht 599451b44SJordan Rupprechtimport lldb 699451b44SJordan Rupprechtfrom lldbsuite.test.decorators import * 799451b44SJordan Rupprechtfrom lldbsuite.test.lldbtest import * 899451b44SJordan Rupprechtfrom lldbsuite.test import lldbutil 999451b44SJordan Rupprecht 1099451b44SJordan Rupprecht 1199451b44SJordan Rupprechtclass StaticVariableTestCase(TestBase): 1299451b44SJordan Rupprecht def setUp(self): 1399451b44SJordan Rupprecht # Call super's setUp(). 1499451b44SJordan Rupprecht TestBase.setUp(self) 1599451b44SJordan Rupprecht # Find the line number to break at. 162238dcc3SJonas Devlieghere self.line = line_number("main.cpp", "// Set break point at this line.") 1799451b44SJordan Rupprecht 1899451b44SJordan Rupprecht def test_with_run_command(self): 1999451b44SJordan Rupprecht """Test that file and class static variables display correctly.""" 2099451b44SJordan Rupprecht self.build() 2199451b44SJordan Rupprecht self.runCmd("file " + self.getBuildArtifact("a.out"), CURRENT_EXECUTABLE_SET) 2299451b44SJordan Rupprecht 2399451b44SJordan Rupprecht lldbutil.run_break_set_by_file_and_line( 242238dcc3SJonas Devlieghere self, "main.cpp", self.line, num_expected_locations=1, loc_exact=True 252238dcc3SJonas Devlieghere ) 2699451b44SJordan Rupprecht 2799451b44SJordan Rupprecht self.runCmd("run", RUN_SUCCEEDED) 2899451b44SJordan Rupprecht 2999451b44SJordan Rupprecht # The stop reason of the thread should be breakpoint. 302238dcc3SJonas Devlieghere self.expect( 312238dcc3SJonas Devlieghere "thread list", 322238dcc3SJonas Devlieghere STOPPED_DUE_TO_BREAKPOINT, 332238dcc3SJonas Devlieghere substrs=["stopped", "stop reason = breakpoint"], 342238dcc3SJonas Devlieghere ) 3599451b44SJordan Rupprecht 3699451b44SJordan Rupprecht # Global variables are no longer displayed with the "frame variable" 3799451b44SJordan Rupprecht # command. 3899451b44SJordan Rupprecht self.expect( 392238dcc3SJonas Devlieghere "target variable A::g_points", 4099451b44SJordan Rupprecht VARIABLES_DISPLAYED_CORRECTLY, 412238dcc3SJonas Devlieghere patterns=["\(PointType\[[1-9]*\]\) A::g_points = {"], 422238dcc3SJonas Devlieghere ) 432238dcc3SJonas Devlieghere self.expect( 442238dcc3SJonas Devlieghere "target variable g_points", 452238dcc3SJonas Devlieghere VARIABLES_DISPLAYED_CORRECTLY, 462238dcc3SJonas Devlieghere substrs=["(PointType[2]) g_points"], 472238dcc3SJonas Devlieghere ) 4899451b44SJordan Rupprecht 4999451b44SJordan Rupprecht # On Mac OS X, gcc 4.2 emits the wrong debug info for A::g_points. 5099451b44SJordan Rupprecht # A::g_points is an array of two elements. 5199451b44SJordan Rupprecht if self.platformIsDarwin() or self.getPlatform() == "linux": 5299451b44SJordan Rupprecht self.expect( 5399451b44SJordan Rupprecht "target variable A::g_points[1].x", 5499451b44SJordan Rupprecht VARIABLES_DISPLAYED_CORRECTLY, 552238dcc3SJonas Devlieghere startstr="(int) A::g_points[1].x = 11", 562238dcc3SJonas Devlieghere ) 5799451b44SJordan Rupprecht 5899451b44SJordan Rupprecht @expectedFailureAll( 592238dcc3SJonas Devlieghere compiler=["gcc"], bugnumber="Compiler emits incomplete debug info" 602238dcc3SJonas Devlieghere ) 6199451b44SJordan Rupprecht @expectedFailureAll( 622238dcc3SJonas Devlieghere compiler=["clang"], compiler_version=["<", "3.9"], bugnumber="llvm.org/pr20550" 632238dcc3SJonas Devlieghere ) 6499451b44SJordan Rupprecht def test_with_run_command_complete(self): 6599451b44SJordan Rupprecht """ 6699451b44SJordan Rupprecht Test that file and class static variables display correctly with 6799451b44SJordan Rupprecht complete debug information. 6899451b44SJordan Rupprecht """ 6999451b44SJordan Rupprecht self.build() 7099451b44SJordan Rupprecht target = self.dbg.CreateTarget(self.getBuildArtifact("a.out")) 7199451b44SJordan Rupprecht self.assertTrue(target, VALID_TARGET) 7299451b44SJordan Rupprecht 7399451b44SJordan Rupprecht # Global variables are no longer displayed with the "frame variable" 7499451b44SJordan Rupprecht # command. 7599451b44SJordan Rupprecht self.expect( 762238dcc3SJonas Devlieghere "target variable A::g_points", 7799451b44SJordan Rupprecht VARIABLES_DISPLAYED_CORRECTLY, 7899451b44SJordan Rupprecht patterns=[ 792238dcc3SJonas Devlieghere "\(PointType\[[1-9]*\]\) A::g_points = {", 802238dcc3SJonas Devlieghere "(x = 1, y = 2)", 812238dcc3SJonas Devlieghere "(x = 11, y = 22)", 822238dcc3SJonas Devlieghere ], 832238dcc3SJonas Devlieghere ) 8499451b44SJordan Rupprecht 8599451b44SJordan Rupprecht # Ensure that we take the context into account and only print 8699451b44SJordan Rupprecht # A::g_points. 8799451b44SJordan Rupprecht self.expect( 882238dcc3SJonas Devlieghere "target variable A::g_points", 8999451b44SJordan Rupprecht VARIABLES_DISPLAYED_CORRECTLY, 9099451b44SJordan Rupprecht matching=False, 912238dcc3SJonas Devlieghere patterns=["(x = 3, y = 4)", "(x = 33, y = 44)"], 922238dcc3SJonas Devlieghere ) 9399451b44SJordan Rupprecht 9499451b44SJordan Rupprecht # Finally, ensure that we print both points when not specifying a 9599451b44SJordan Rupprecht # context. 9699451b44SJordan Rupprecht self.expect( 972238dcc3SJonas Devlieghere "target variable g_points", 9899451b44SJordan Rupprecht VARIABLES_DISPLAYED_CORRECTLY, 9999451b44SJordan Rupprecht substrs=[ 1002238dcc3SJonas Devlieghere "(PointType[2]) g_points", 1012238dcc3SJonas Devlieghere "(x = 1, y = 2)", 1022238dcc3SJonas Devlieghere "(x = 11, y = 22)", 1032238dcc3SJonas Devlieghere "(x = 3, y = 4)", 1042238dcc3SJonas Devlieghere "(x = 33, y = 44)", 1052238dcc3SJonas Devlieghere ], 1062238dcc3SJonas Devlieghere ) 10799451b44SJordan Rupprecht 10822667e32SJim Ingham def build_value_check(self, var_name, values): 109ab855530SJonas Devlieghere children_1 = [ 110ab855530SJonas Devlieghere ValueCheck(name="x", value=values[0], type="int"), 111ab855530SJonas Devlieghere ValueCheck(name="y", value=values[1], type="int"), 112ab855530SJonas Devlieghere ] 113ab855530SJonas Devlieghere children_2 = [ 114ab855530SJonas Devlieghere ValueCheck(name="x", value=values[2], type="int"), 115ab855530SJonas Devlieghere ValueCheck(name="y", value=values[3], type="int"), 116ab855530SJonas Devlieghere ] 117ab855530SJonas Devlieghere elem_0 = ValueCheck( 118ab855530SJonas Devlieghere name="[0]", value=None, type="PointType", children=children_1 119ab855530SJonas Devlieghere ) 120ab855530SJonas Devlieghere elem_1 = ValueCheck( 121ab855530SJonas Devlieghere name="[1]", value=None, type="PointType", children=children_2 122ab855530SJonas Devlieghere ) 123ab855530SJonas Devlieghere value_check = ValueCheck( 124ab855530SJonas Devlieghere name=var_name, value=None, type="PointType[2]", children=[elem_0, elem_1] 125ab855530SJonas Devlieghere ) 12622667e32SJim Ingham 12722667e32SJim Ingham return value_check 12822667e32SJim Ingham 12999451b44SJordan Rupprecht @expectedFailureAll( 1302238dcc3SJonas Devlieghere compiler=["gcc"], bugnumber="Compiler emits incomplete debug info" 1312238dcc3SJonas Devlieghere ) 13299451b44SJordan Rupprecht @expectedFailureAll( 1332238dcc3SJonas Devlieghere compiler=["clang"], compiler_version=["<", "3.9"], bugnumber="llvm.org/pr20550" 1342238dcc3SJonas Devlieghere ) 13599451b44SJordan Rupprecht @expectedFailureAll(oslist=["windows"], bugnumber="llvm.org/pr24764") 1362238dcc3SJonas Devlieghere @add_test_categories(["pyapi"]) 137df1bb2e6SJim Ingham def test_with_python_FindValue(self): 13899451b44SJordan Rupprecht """Test Python APIs on file and class static variables.""" 13999451b44SJordan Rupprecht self.build() 14099451b44SJordan Rupprecht exe = self.getBuildArtifact("a.out") 14199451b44SJordan Rupprecht 14299451b44SJordan Rupprecht target = self.dbg.CreateTarget(exe) 14399451b44SJordan Rupprecht self.assertTrue(target, VALID_TARGET) 14499451b44SJordan Rupprecht 14599451b44SJordan Rupprecht breakpoint = target.BreakpointCreateByLocation("main.cpp", self.line) 14699451b44SJordan Rupprecht self.assertTrue(breakpoint, VALID_BREAKPOINT) 14799451b44SJordan Rupprecht 14899451b44SJordan Rupprecht # Now launch the process, and do not stop at entry point. 1492238dcc3SJonas Devlieghere process = target.LaunchSimple(None, None, self.get_process_working_directory()) 15099451b44SJordan Rupprecht self.assertTrue(process, PROCESS_IS_VALID) 15199451b44SJordan Rupprecht 15299451b44SJordan Rupprecht # The stop reason of the thread should be breakpoint. 1532238dcc3SJonas Devlieghere thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint) 15499451b44SJordan Rupprecht self.assertIsNotNone(thread) 15599451b44SJordan Rupprecht 15699451b44SJordan Rupprecht # Get the SBValue of 'A::g_points' and 'g_points'. 15799451b44SJordan Rupprecht frame = thread.GetFrameAtIndex(0) 15899451b44SJordan Rupprecht 15999451b44SJordan Rupprecht # arguments => False 16099451b44SJordan Rupprecht # locals => False 16199451b44SJordan Rupprecht # statics => True 16299451b44SJordan Rupprecht # in_scope_only => False 16399451b44SJordan Rupprecht valList = frame.GetVariables(False, False, True, False) 16499451b44SJordan Rupprecht 16522667e32SJim Ingham # Build ValueCheckers for the values we're going to find: 16622667e32SJim Ingham value_check_A = self.build_value_check("A::g_points", ["1", "2", "11", "22"]) 16722667e32SJim Ingham value_check_none = self.build_value_check("g_points", ["3", "4", "33", "44"]) 16822667e32SJim Ingham value_check_AA = self.build_value_check("AA::g_points", ["5", "6", "55", "66"]) 16922667e32SJim Ingham 17099451b44SJordan Rupprecht for val in valList: 17199451b44SJordan Rupprecht self.DebugSBValue(val) 17299451b44SJordan Rupprecht name = val.GetName() 17322667e32SJim Ingham self.assertIn(name, ["g_points", "A::g_points", "AA::g_points"]) 17422667e32SJim Ingham 17522667e32SJim Ingham if name == "A::g_points": 17622667e32SJim Ingham self.assertEqual(val.GetValueType(), lldb.eValueTypeVariableGlobal) 17722667e32SJim Ingham value_check_A.check_value(self, val, "Got A::g_points right") 1782238dcc3SJonas Devlieghere if name == "g_points": 1792238dcc3SJonas Devlieghere self.assertEqual(val.GetValueType(), lldb.eValueTypeVariableStatic) 18022667e32SJim Ingham value_check_none.check_value(self, val, "Got g_points right") 18122667e32SJim Ingham if name == "AA::g_points": 1822238dcc3SJonas Devlieghere self.assertEqual(val.GetValueType(), lldb.eValueTypeVariableGlobal) 18322667e32SJim Ingham value_check_AA.check_value(self, val, "Got AA::g_points right") 18499451b44SJordan Rupprecht 18599451b44SJordan Rupprecht # SBFrame.FindValue() should also work. 18699451b44SJordan Rupprecht val = frame.FindValue("A::g_points", lldb.eValueTypeVariableGlobal) 18799451b44SJordan Rupprecht self.DebugSBValue(val) 18822667e32SJim Ingham value_check_A.check_value(self, val, "FindValue also works") 18999451b44SJordan Rupprecht 19099451b44SJordan Rupprecht # Also exercise the "parameter" and "local" scopes while we are at it. 19199451b44SJordan Rupprecht val = frame.FindValue("argc", lldb.eValueTypeVariableArgument) 19299451b44SJordan Rupprecht self.DebugSBValue(val) 1932238dcc3SJonas Devlieghere self.assertEqual(val.GetName(), "argc") 19499451b44SJordan Rupprecht 19599451b44SJordan Rupprecht val = frame.FindValue("argv", lldb.eValueTypeVariableArgument) 19699451b44SJordan Rupprecht self.DebugSBValue(val) 1972238dcc3SJonas Devlieghere self.assertEqual(val.GetName(), "argv") 19899451b44SJordan Rupprecht 19999451b44SJordan Rupprecht val = frame.FindValue("hello_world", lldb.eValueTypeVariableLocal) 20099451b44SJordan Rupprecht self.DebugSBValue(val) 2012238dcc3SJonas Devlieghere self.assertEqual(val.GetName(), "hello_world") 20222667e32SJim Ingham 203df1bb2e6SJim Ingham # This test tests behavior that's been broken for a very long time.. 204df1bb2e6SJim Ingham # The fix for it is in the accelerator table part of the DWARF reader, 205df1bb2e6SJim Ingham # and I fixed the version that the names accelerator uses, but I don't 206df1bb2e6SJim Ingham # know how to fix it on systems that don't use that. There isn't a 207df1bb2e6SJim Ingham # decorator for that - not sure how to construct that so I'm limiting the 208df1bb2e6SJim Ingham # test do Darwin for now. 209df1bb2e6SJim Ingham @expectedFailureAll( 210df1bb2e6SJim Ingham compiler=["gcc"], bugnumber="Compiler emits incomplete debug info" 211df1bb2e6SJim Ingham ) 212df1bb2e6SJim Ingham @skipUnlessDarwin 213df1bb2e6SJim Ingham @add_test_categories(["pyapi"]) 214df1bb2e6SJim Ingham def test_with_python_FindGlobalVariables(self): 215df1bb2e6SJim Ingham """Test Python APIs on file and class static variables.""" 216df1bb2e6SJim Ingham self.build() 217df1bb2e6SJim Ingham exe = self.getBuildArtifact("a.out") 218df1bb2e6SJim Ingham 219df1bb2e6SJim Ingham target = self.dbg.CreateTarget(exe) 220df1bb2e6SJim Ingham self.assertTrue(target, VALID_TARGET) 221df1bb2e6SJim Ingham 222df1bb2e6SJim Ingham breakpoint = target.BreakpointCreateByLocation("main.cpp", self.line) 223df1bb2e6SJim Ingham self.assertTrue(breakpoint, VALID_BREAKPOINT) 224df1bb2e6SJim Ingham 225df1bb2e6SJim Ingham # Now launch the process, and do not stop at entry point. 226df1bb2e6SJim Ingham process = target.LaunchSimple(None, None, self.get_process_working_directory()) 227df1bb2e6SJim Ingham self.assertTrue(process, PROCESS_IS_VALID) 228df1bb2e6SJim Ingham 229df1bb2e6SJim Ingham # The stop reason of the thread should be breakpoint. 230df1bb2e6SJim Ingham thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint) 231df1bb2e6SJim Ingham self.assertIsNotNone(thread) 232df1bb2e6SJim Ingham 233df1bb2e6SJim Ingham # Get the SBValue of 'A::g_points' and 'g_points'. 234df1bb2e6SJim Ingham frame = thread.GetFrameAtIndex(0) 235df1bb2e6SJim Ingham 236df1bb2e6SJim Ingham # Build ValueCheckers for the values we're going to find: 237df1bb2e6SJim Ingham value_check_A = self.build_value_check("A::g_points", ["1", "2", "11", "22"]) 238df1bb2e6SJim Ingham value_check_none = self.build_value_check("g_points", ["3", "4", "33", "44"]) 239df1bb2e6SJim Ingham value_check_AA = self.build_value_check("AA::g_points", ["5", "6", "55", "66"]) 240df1bb2e6SJim Ingham 24122667e32SJim Ingham # We should also be able to get class statics from FindGlobalVariables. 24222667e32SJim Ingham # eMatchTypeStartsWith should only find A:: not AA:: 24322667e32SJim Ingham val_list = target.FindGlobalVariables("A::", 10, lldb.eMatchTypeStartsWith) 24422667e32SJim Ingham self.assertEqual(val_list.GetSize(), 1, "Found only one match") 24522667e32SJim Ingham val = val_list[0] 24622667e32SJim Ingham value_check_A.check_value(self, val, "FindGlobalVariables starts with") 24722667e32SJim Ingham 24822667e32SJim Ingham # Regex should find both 24922667e32SJim Ingham val_list = target.FindGlobalVariables("A::", 10, lldb.eMatchTypeRegex) 25022667e32SJim Ingham self.assertEqual(val_list.GetSize(), 2, "Found A & AA") 25122667e32SJim Ingham found_a = False 25222667e32SJim Ingham found_aa = False 25322667e32SJim Ingham for val in val_list: 25422667e32SJim Ingham name = val.GetName() 25522667e32SJim Ingham if name == "A::g_points": 25622667e32SJim Ingham value_check_A.check_value(self, val, "AA found by regex") 25722667e32SJim Ingham found_a = True 25822667e32SJim Ingham elif name == "AA::g_points": 25922667e32SJim Ingham value_check_AA.check_value(self, val, "A found by regex") 26022667e32SJim Ingham found_aa = True 26122667e32SJim Ingham 26222667e32SJim Ingham self.assertTrue(found_a, "Regex search found A::g_points") 26322667e32SJim Ingham self.assertTrue(found_aa, "Regex search found AA::g_points") 26422667e32SJim Ingham 265*8f2a4e83SJonas Devlieghere # Regex lowercase should find both as well. 266*8f2a4e83SJonas Devlieghere val_list = target.FindGlobalVariables( 267*8f2a4e83SJonas Devlieghere "a::g_points", 10, lldb.eMatchTypeRegexInsensitive 268*8f2a4e83SJonas Devlieghere ) 269*8f2a4e83SJonas Devlieghere self.assertEqual(val_list.GetSize(), 2, "Found A & AA") 270*8f2a4e83SJonas Devlieghere 27122667e32SJim Ingham # Normal search for full name should find one, but it looks like we don't match 27222667e32SJim Ingham # on identifier boundaries here yet: 27322667e32SJim Ingham val_list = target.FindGlobalVariables("A::g_points", 10, lldb.eMatchTypeNormal) 274ab855530SJonas Devlieghere self.assertEqual( 275ab855530SJonas Devlieghere val_list.GetSize(), 2, "We aren't matching on name boundaries yet" 276ab855530SJonas Devlieghere ) 27722667e32SJim Ingham 27822667e32SJim Ingham # Normal search for g_points should find 3 - FindGlobalVariables doesn't distinguish 27922667e32SJim Ingham # between file statics and globals: 28022667e32SJim Ingham val_list = target.FindGlobalVariables("g_points", 10, lldb.eMatchTypeNormal) 28122667e32SJim Ingham self.assertEqual(val_list.GetSize(), 3, "Found all three g_points") 282