Index: lldb/packages/Python/lldbsuite/test/lldbtest.py =================================================================== --- lldb/packages/Python/lldbsuite/test/lldbtest.py +++ lldb/packages/Python/lldbsuite/test/lldbtest.py @@ -2442,12 +2442,87 @@ msg + "\nCommand output:\n" + EXP_MSG(str, output, exe) if msg else EXP_MSG(str, output, exe)) + class ValueCheck: + def __init__(self, expect_name=None, expect_value=None, expect_type=None, + expect_summary=None, children=None): + """ + :param expect_name: The name that the SBValue should have. None if + the summary should not be checked. + :param expect_summary: The summary that the SBValue should have. + None if the summary should not be checked. + :param expect_value: The value that the SBValue should have. + None if the value should not be checked. + :param expect_type: The type that the SBValue result should have. + None if the type should not be checked. + :param children: A list of ValueChecks that need to match the + children of this SBValue. None if children + shouldn't be checked. The order of checks is the + order of the checks in the list. The number of + checks has to match the number of children. + """ + self.expect_name = expect_name + self.expect_value = expect_value + self.expect_type = expect_type + self.expect_summary = expect_summary + self.children = children + + def check_value(self, test_base, val, error_msg=None): + """ + Checks that the given value matches the currently set properties + of this ValueCheck. If a match failed, the given TestBase will + be used to emit an error. A custom error message can be specified + that will be used to describe failed check for this SBValue (but + not errors in the child values). + """ + + this_error_msg = error_msg if error_msg else "" + this_error_msg += "\nChecking SBValue: " + str(val) + + test_base.assertSuccess(val.GetError()) + + if self.expect_name: + test_base.assertEqual(self.expect_name, val.GetName(), + this_error_msg) + if self.expect_value: + test_base.assertEqual(self.expect_value, val.GetValue(), + this_error_msg) + if self.expect_type: + test_base.assertEqual(self.expect_type, val.GetDisplayTypeName(), + this_error_msg) + if self.expect_summary: + test_base.assertEqual(self.expect_summary, val.GetSummary(), + this_error_msg) + if self.children is not None: + test_base.check_value_children(val, self.children, error_msg) + + def check_value_children(self, val, expected_children, error_msg=None): + """ + Checks that the children of a SBValue match a certain structure and + have certain properties. + + :param val: The SBValue to check. + :param expected_children: A list of ValueChecks. The order and number of + checks has to match the order of children in + the given value. + """ + + this_error_msg = error_msg if error_msg else "" + this_error_msg += "\nChecking SBValue: " + str(val) + + self.assertEqual(len(expected_children), val.GetNumChildren(), this_error_msg) + + for i in range(0, val.GetNumChildren()): + expected_child = expected_children[i] + actual_child = val.GetChildAtIndex(i) + expected_child.check_value(self, actual_child, error_msg) + def expect_expr( self, expr, result_summary=None, result_value=None, result_type=None, + result_children=None ): """ Evaluates the given expression and verifies the result. @@ -2455,6 +2530,8 @@ :param result_summary: The summary that the expression should have. None if the summary should not be checked. :param result_value: The value that the expression should have. None if the value should not be checked. :param result_type: The type that the expression result should have. None if the type should not be checked. + :param result_children: The expected children of the expression result + as a list of ValueChecks. None if the children shouldn't be checked. """ self.assertTrue(expr.strip() == expr, "Expression contains trailing/leading whitespace: '" + expr + "'") @@ -2473,16 +2550,11 @@ else: eval_result = self.target().EvaluateExpression(expr, options) - self.assertSuccess(eval_result.GetError()) - - if result_type: - self.assertEqual(result_type, eval_result.GetDisplayTypeName()) - - if result_value: - self.assertEqual(result_value, eval_result.GetValue()) - - if result_summary: - self.assertEqual(result_summary, eval_result.GetSummary()) + value_check = self.ValueCheck(expect_type=result_type, + expect_value=result_value, + expect_summary=result_summary, + children=result_children) + value_check.check_value(self, eval_result, str(eval_result)) def invoke(self, obj, name, trace=False): """Use reflection to call a method dynamically with no argument.""" Index: lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/map/TestDataFormatterLibccMap.py =================================================================== --- lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/map/TestDataFormatterLibccMap.py +++ lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/map/TestDataFormatterLibccMap.py @@ -19,6 +19,11 @@ ns = 'ndk' if lldbplatformutil.target_is_android() else '' self.namespace = 'std' + def check_pair(self, first_value, second_value): + pair_children = [self.ValueCheck(expect_name="first", expect_value=first_value), + self.ValueCheck(expect_name="second", expect_value=second_value)] + return self.ValueCheck(children=pair_children) + @add_test_categories(["libc++"]) def test_with_run_command(self): """Test that that file and class static variables display correctly.""" @@ -51,10 +56,8 @@ self.addTearDownHook(cleanup) ns = self.namespace - self.expect('p ii', - substrs=['%s::map' % ns, - 'size=0', - '{}']) + self.expect_expr("ii", result_summary="size=0", result_children=[]) + self.expect('frame var ii', substrs=['%s::map' % ns, 'size=0', @@ -62,14 +65,10 @@ lldbutil.continue_to_breakpoint(self.process(), bkpt) - self.expect('p ii', - substrs=['%s::map' % ns, 'size=2', - '[0] = ', - 'first = 0', - 'second = 0', - '[1] = ', - 'first = 1', - 'second = 1']) + self.expect_expr("ii", result_summary="size=2", result_children=[ + self.check_pair("0", "0"), + self.check_pair("1", "1") + ]) self.expect('frame variable ii', substrs=['%s::map' % ns, 'size=2', Index: lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/vector/TestDataFormatterLibcxxVector.py =================================================================== --- lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/vector/TestDataFormatterLibcxxVector.py +++ lldb/test/API/functionalities/data-formatter/data-formatter-stl/libcxx/vector/TestDataFormatterLibcxxVector.py @@ -26,16 +26,15 @@ '[6] = 1234567', '}']) - self.expect("p " + var_name, - substrs=['$', 'size=7', - '[0] = 1', - '[1] = 12', - '[2] = 123', - '[3] = 1234', - '[4] = 12345', - '[5] = 123456', - '[6] = 1234567', - '}']) + self.expect_expr(var_name, result_summary="size=7", result_children=[ + self.ValueCheck(expect_value="1"), + self.ValueCheck(expect_value="12"), + self.ValueCheck(expect_value="123"), + self.ValueCheck(expect_value="1234"), + self.ValueCheck(expect_value="12345"), + self.ValueCheck(expect_value="123456"), + self.ValueCheck(expect_value="1234567"), + ]) # check access-by-index self.expect("frame variable " + var_name + "[0]",