blob: 800c591def7c884003ddfa26f9ce5d9bc351801a [file]
"""
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)
@expectedFailureAll(
oslist=["windows"],
bugnumber="llvm.org/pr21765",
)
@add_test_categories(["pyapi"])
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)
func_call = "(int)getpid()"
if lldbplatformutil.getPlatform() == "windows":
func_call = "(int)GetCurrentProcessId()"
for expression in expressions:
interp_expression = expression
jit_expression = func_call + "; " + expression
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)
def test_fpconv(self):
self.build_and_run()
interp_options = lldb.SBExpressionOptions()
interp_options.SetLanguage(lldb.eLanguageTypeC_plus_plus)
interp_options.SetAllowJIT(False)
jit_options = lldb.SBExpressionOptions()
jit_options.SetLanguage(lldb.eLanguageTypeC_plus_plus)
jit_options.SetAllowJIT(True)
set_up_expressions = [
"int32_t $i = 3",
"int32_t $n = -3",
"uint32_t $u = 5",
"int64_t $l = -7",
"float $f = 9.0625",
"double $d = 13.75",
"float $nf = -11.25",
]
expressions = [
"$i + $f", # sitofp i32 to float
"$d - $n", # sitofp i32 to double
"$u + $f", # uitofp i32 to float
"$u + $d", # uitofp i32 to double
"(int32_t)$d", # fptosi double to i32
"(int32_t)$f", # fptosi float to i32
"(int64_t)$d", # fptosi double to i64
"(int16_t)$f", # fptosi float to i16
"(int64_t)$nf", # fptosi float to i64
"(uint16_t)$f", # fptoui float to i16
"(uint32_t)$d", # fptoui double to i32
"(uint64_t)$d", # fptoui double to i64
"(float)$d", # fptrunc double to float
"(double)$f", # fpext float to double
"(double)$nf", # fpext float to double
]
for expression in set_up_expressions:
self.frame().EvaluateExpression(expression, interp_options)
func_call = "(int)getpid()"
if lldbplatformutil.getPlatform() == "windows":
func_call = "(int)GetCurrentProcessId()"
for expression in expressions:
interp_expression = expression
# Calling a function forces the expression to be executed with JIT.
jit_expression = func_call + "; " + expression
interp_result = self.frame().EvaluateExpression(
interp_expression, interp_options
)
jit_result = self.frame().EvaluateExpression(jit_expression, jit_options)
self.assertEqual(
interp_result.GetValue(),
jit_result.GetValue(),
"Values match for " + expression,
)
self.assertEqual(
interp_result.GetTypeName(),
jit_result.GetTypeName(),
"Types match for " + expression,
)
def test_fpconv_ub(self):
target = self.dbg.GetDummyTarget()
set_up_expressions = [
"float $f = 3e9",
"double $d = 1e20",
"float $nf = -1.5",
]
expressions = [
(
"(int32_t)$f",
"Conversion error: (float) 3.0E+9 cannot be converted to i32",
),
(
"(uint32_t)$nf",
"Conversion error: (float) -1.5 cannot be converted to unsigned i32",
),
(
"(int64_t)$d",
"Conversion error: (float) 1.0E+20 cannot be converted to i64",
),
(
"(uint64_t)$d",
"Conversion error: (float) 1.0E+20 cannot be converted to unsigned i64",
),
]
for expression in set_up_expressions:
target.EvaluateExpression(expression)
# The IR Interpreter returns an error if a value cannot be converted.
for expression in expressions:
result = target.EvaluateExpression(expression[0])
self.assertIn(expression[1], str(result.GetError()))
# The conversion should succeed if the destination type can represent the result.
self.expect_expr(
"(uint32_t)$f", result_type="uint32_t", result_value="3000000000"
)
self.expect_expr("(int32_t)$nf", result_type="int32_t", result_value="-1")