""" Test the IR interpreter """ import lldb from lldbsuite.test.decorators import * from lldbsuite.test.lldbtest import * from lldbsuite.test import lldbutil class IRInterpreterTestCase(TestBase): NO_DEBUG_INFO_TESTCASE = True def time_expression(self, expr, options): start = time.time() res = self.target.EvaluateExpression(expr, options) return res, time.time() - start def test_interpreter_timeout(self): """Test the timeout logic in the IRInterpreter.""" self.build() self.target = self.dbg.CreateTarget(self.getBuildArtifact("a.out")) self.assertTrue(self.target, VALID_TARGET) # A non-trivial infinite loop. inf_loop = "for (unsigned i = 0; i < 100; ++i) --i; 1" timeout_error = "Reached timeout while interpreting expression" options = lldb.SBExpressionOptions() # This is an IRInterpreter specific test, so disable the JIT. options.SetAllowJIT(False) # We use a 500ms timeout. options.SetTimeoutInMicroSeconds(500000) res, duration_sec = self.time_expression(inf_loop, options) self.assertIn(timeout_error, str(res.GetError())) # Depending on the machine load the expression might take quite some # time, so give the time a generous upper bound. self.assertLess(duration_sec, 15) # Try a simple one second timeout. options.SetTimeoutInMicroSeconds(1000000) res, duration_sec = self.time_expression(inf_loop, options) self.assertIn(timeout_error, str(res.GetError())) # Anything within 5% of 1s is fine, to account for machine differences. self.assertGreaterEqual(duration_sec, 0.95) self.assertLess(duration_sec, 30) def test_interpreter_interrupt(self): """Test interrupting the IRInterpreter.""" self.build() self.target = self.dbg.CreateTarget(self.getBuildArtifact("a.out")) self.assertTrue(self.target, VALID_TARGET) # A non-trivial infinite loop. inf_loop = "for (unsigned i = 0; i < 100; ++i) --i; 1" options = lldb.SBExpressionOptions() # This is an IRInterpreter specific test, so disable the JIT. options.SetAllowJIT(False) # Make sure we have a pretty long (10s) timeout so we have a chance to # interrupt the interpreted expression. options.SetTimeoutInMicroSeconds(10000000) self.dbg.RequestInterrupt() self.dbg.SetAsync(True) res = self.target.EvaluateExpression(inf_loop, options) self.dbg.SetAsync(False) # Be sure to turn this off again: def cleanup(): if self.dbg.InterruptRequested(): self.dbg.CancelInterruptRequest() self.addTearDownHook(cleanup) interrupt_error = "Interrupted while interpreting expression" self.assertIn(interrupt_error, str(res.GetError())) def setUp(self): # Call super's setUp(). TestBase.setUp(self) # Find the line number to break for main.c. self.line = line_number("main.c", "// Set breakpoint here") # Disable confirmation prompt to avoid infinite wait self.runCmd("settings set auto-confirm true") self.addTearDownHook(lambda: self.runCmd("settings clear auto-confirm")) def build_and_run(self): """Test the IR interpreter""" self.build() self.runCmd("file " + self.getBuildArtifact("a.out"), CURRENT_EXECUTABLE_SET) lldbutil.run_break_set_by_file_and_line( self, "main.c", self.line, num_expected_locations=1, loc_exact=False ) self.runCmd("run", RUN_SUCCEEDED) @add_test_categories(["pyapi"]) # getpid() is POSIX, among other problems, see bug @expectedFailureAll(oslist=["windows"], bugnumber="http://llvm.org/pr21765") def test_ir_interpreter(self): self.build_and_run() options = lldb.SBExpressionOptions() options.SetLanguage(lldb.eLanguageTypeC_plus_plus) set_up_expressions = [ "int $i = 9", "int $j = 3", "int $k = 5", "unsigned long long $ull = -1", "unsigned $u = -1", ] expressions = [ "$i + $j", "$i - $j", "$i * $j", "$i / $j", "$i % $k", "$i << $j", "$i & $j", "$i | $j", "$i ^ $j", "($ull & -1) == $u", ] for expression in set_up_expressions: self.frame().EvaluateExpression(expression, options) for expression in expressions: interp_expression = expression jit_expression = "(int)getpid(); " + expression interp_result = ( self.frame() .EvaluateExpression(interp_expression, options) .GetValueAsSigned() ) jit_result = ( self.frame() .EvaluateExpression(jit_expression, options) .GetValueAsSigned() ) self.assertEqual( interp_result, jit_result, "While evaluating " + expression ) def test_type_conversions(self): target = self.dbg.GetDummyTarget() short_val = target.EvaluateExpression("(short)-1") self.assertEqual(short_val.GetValueAsSigned(), -1) long_val = target.EvaluateExpression("(long) " + short_val.GetName()) self.assertEqual(long_val.GetValueAsSigned(), -1)